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西 泽 梦 路 


生 于 丧 玉 县 。 技 术 写 作 人 ， 同 时 
也 是 一 名 大 学 研究 员 ， 主 要 研究 领域 
为 信息 教育 。 著 有 《 Oracle 基础 教 
程 》《PHP 基 础 教程 》 等 。 
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Oracle 数 据 库 技术 支持 工程 师 。 
Oracle 全 球 客户 支持 ( GCS 大 连 ) 数 
据 库 性 能 与 安全 日 语 组 组 长 ， 中 文 
Tier1 工 程 师 。 
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效 字 有 版权 声明 


图 灵 社 区 的 电子 书 没有 采用 专 有 客 
户 端 ， 您 可 以 在 任意 设备 上 ， 用 自 
己 喜 欢 的 浏览 器 和 PDF 阅读 器 进行 
阅读 。 

但 您 购买 的 电子 书 仅 供 您 个 人 使 用 ， 
未 经 授权 ， 不 得 进行 传播 。 

我 们 愿意 相信 读者 具有 这 样 的 良知 
和 觉悟 ， 与 我 们 共同 保护 知识 产权 。 


如 果 购 买 者 有 侵权 行为 ， 我 们 可 能 
对 该 用 户 实施 包括 但 不 限于 关闭 该 
帐号 等 维权 措施 ， 并 可 能 追究 法 律 
责任 。 
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本 书 介绍 了 MySQL 的 操作 方法 以 及 通过 使 用 PHP 和 MySQL 创建 Web 应 用 程序 的 基础 知识 。 作 者 
从 数据 库 是 什么 开始 讲 起 ， 由 浅 入 深 ,通过 丰富 的 图 示 和 大 量 的 示例 程序 ， 让 读者 循序 渐进 地 掌握 
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相信 许多 人 有 过 这 样 的 经 历 : 想 学 做 一 道 菜 ， 于 是 深 研 荣 谱 ， 知 道 了 如 何 搭配 食材 ， 如 何 控制 
火候 ， 选 取 多 少 分 量 ， 但 最 终 还 是 没有 做 出 自己 理想 中 的 完美 菜肴 。 

如 果 有 一 位 厨 艺 精湛 的 师傅 手把手 传授 做 菜 技巧 ， 帮 助 你 理解 并 领悟 其 中 的 精 散 ， 相 信 经 过 时 
间 的 沉 洗 ， 新 手 也 能 成 为 真正 的 大 厨 ! 

一 本 好 书 正如 一 位 好 老师 一 一 传道 、 授 业 、 解 惑 。 它 不 仅 能 系统 地 讲解 知识 ， 还 能 引导 学 生动 
手 实践 ， 让 学 生 在 学 和 做 的 交互 过 程 中 巩固 理论 知识 ， 掌 握 技术 技巧 。 

撰写 本 书 的 意义 正在 于 此 。 作 为 MySQL 的 入 门 教程 ， 作 者 充分 利用 了 自己 在 相关 领域 积累 的 
经 验 ， 以 深入 浅 出 的 方式 进行 了 阐述 ， 并 且 在 相关 章节 中 插入 了 大 量 的 实践 代码 和 图 表 作为 补充 ， 
使 读者 能 够 毫 不 费力 地 跟随 作者 的 思路 ， 系 统 全 面 地 掌握 相应 的 理论 知识 和 技术 技巧 。 

本 书 介绍 的 MySQL 相关 知识 由 以 下 6 个 部 分 组 成 。 













































































@ 第 1 部 分 初 识 MySQL 
第 1 部 分 有 2 章 内 容 ， 介绍 了 人 MySQL 和 数据 库 的 概要 ， 以 及 如 何 通过 MAMP 软件 快速 构建 
MySQL+Apache+PHP 运行 环境 。 























@ 第 2 部 分 MySQL 的 基础 知识 
第 2 部 分 有 5 章 内 容 ， 主 要 介绍 了 MySQL 的 基础 知识 ， 其 中 包括 通过 MySQL 监视 器 创 建 数 
据 库 ， 创 建 表 ， 插 和 人 数据 ， 确 认 数据 ， 以 及 修改 、 复 制 和 删除 表 等 基础 操作 。 











@ 第 3 部 分 熟练 使 用 MySQL 
第 3 部 分 有 7 章 内 容 ,介绍 了 MySQL 的 一 些 复 杂 操 作 ， 比 如 使 用 各 种 条 件 进行 提取 、 编 辑 
等 ， 还 介绍 了 视图 、 存 储 过 程 、 事 务 和 文件 操作 的 方法 等 内 容 。 























@ 第 4 部 分 MySQL+PHP 的 基础 
第 4 部 分 有 4 章 内 容 ， 介绍 了 运用 MySQL 时 必须 掌握 的 Web 和 PHP 知识 ， 以 及 使 用 PHP 脚 
本 操作 MySQL 的 方法 等 。 





@ 第 5 部 分 MySQL + PHP 的 实践 
第 5 部 分 有 3 章 内 容 ， 介 绍 了 如 何 运用 所 学 知识 创建 一 个 实用 的 公告 板 系统 。 














@ 第 6 部 分 附录 
第 6 部 分 包括 phpMyAdmin 的 使 用 方法 、 常 见 问题 的 检查 清单 ， 以 及 MySQL 基础 练习 。 











正如 老师 课 后 布置 作业 以 帮助 学 生 消化 和 理解 学 习 内 容 一 样 ， 
进行 了 总 结 ， 并 精心 设计 了 练习 题 和 参考 答案 ， 以 帮助 读者 自我 检 
作者 在 以 通俗 易 懂 的 方式 介绍 基础 知识 的 同时 ， 还 分 享 了 许多 实用 的 技巧 。 比 如 5.4.2 节 介 绍 
了 通过 “- -prompt = 提示 符 内 容 ” 选 项 设置 提示 符 ， 在 提示 符 “>” 中 指定 自己 想 显 示 的 内 容 ; 


10.4.2 节 和 10.5.8 节 介 绍 了 不 同 的 提 

















础 知识 ， 掌 握 很 多 能 够 立刻 在 工作 中 使 用 的 硬 技能 。 





由 于 译 者 水 平 有 限 ， 书 中 怒 有 琉 漏 之 处 ， 还 望 读者 不 音 赐 教 。 











感谢 我 的 同事 、 我 的 朋友 一 一 高 级 解决 方案 架构 昨 








作者 在 每 章 结尾 也 对 相关 知识 点 
测 ， 巩固 知 识 点 。 























F 序 技巧 ， 等 等 。 这 些 实用 的 技巧 能 够 帮助 读者 巩固 MySQL 基 


i 张 馈 和 售 前 技术 支持 顾问 陈 以 根 帮 我 审阅 本 

















书 的 译 稿 并 给 出 诸多 宝贵 的 意见 和 建议 。 感 谢 图 灵 公 司 的 各 位 编辑 在 翻译 过 程 中 给 予 的 帮助 和 指 
导 。 感 谢 我 的 家 人 给 予 的 支持 。 正 因为 有 你 们 的 帮助 和 支持 ， 我 才 
希望 本 书 能 够 起 到 抛砖引玉 的 作用 ， 帮 助 更 多 的 读者 开启 MySQL 数据 库 的 学 习 之 路 。 和 希望 读 
者 能 以 此 为 基础 不 断 学 习 和 和 销 研 ， 形 成 自己 的 一 套 知识 体系 ， 获 取经 验 ， 取 得 进步 1 
同时 ， 欢 迎 大 家 关注 我 的 公众 号 ( Oracle 数据 库 技术 : TeacherWhat ) 进行 学 习 和 交流 。 


























田 Windows 是 微软 公司 在 美国 和 其 他 国家 的 注册 商标 。 


加 本 
国 本 
国 本 











上 中 出 现 的 系统 、 商 品名 分 别 是 各 公司 的 商标 及 注册 商标 。 
中 没有 明确 标记 TM 商标 和 及 商标 。 
中 出 现 的 网 址 可 能 会 发 生变 更 。 











能 完成 本 书 的 翻译 。 








卢 克 
2019 年 10 月 于 大 连 


中 








© 2017 本 书 内 容 受 著作 权 法 保护 。 


未 经 作者 及 出 版 方 的 允许 ,禁止 复制 、 转 载 本 书 内 容 。 
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“ 星 涩 难 懂 地 解释 复杂 的 东西 ”很 简单 ,“ 浅 显 易 懂 地 解释 复杂 的 东西 ” 却 非常 困难 。 以 这 句 话 
开篇 的 本 书 ， 首 次 出 版 已 经 是 2007 年 的 事 了 。 

我 至 今 依然 记得 在 本 书 的 策划 阶段 ， 编 辑 部 的 编辑 提出 了 “绝对 不 接受 让 读者 似 懂 非 慌 的 
书 ”“ 书 中 示例 涉及 的 知识 一 定 要 毫 无 遗漏 地 进行 解说 ”等 要 求 。 还 记得 在 交 稿 前 一 天 的 深夜 ， 直 
到 最 后 时 刻 我 还 在 和 编辑 人 员 对 原稿 进行 修改 。 这 一 切 仿佛 发 生 在 昨天 。 

承蒙 大 家 的 厚爱 ,《MySQL 基础 教程 》 不 断 加 印 ， 还 发 行 了 修订 版 、 电 子 版 ， 并 且 收 到 了 非常 
好 的 评价 ， 从 诞生 到 现在 已 经 有 10 年 了 。 另 外 ， 本 书 韩文 版 也 已 出 版 并 加 印 ， 作 为 作者 ， 我 感到 
非常 荣幸 。 当 然 和 10 年 前 相 比 ，MySQL 的 相关 情况 发 生 了 巨大 的 变化 ， 所 以 为 了 适应 新 时 代 的 要 
求 ， 我 更 新 了 本 书 中 MySQL 和 PHP 的 版 本 ， 并 进行 了 相关 说 明 ， 希望 能 让 本 书 更 易于 理解 。 

本 书 介绍 了 MySQL 的 操作 方法 以 及 通过 使 用 PHP 和 MySQL 创建 Web 应 用 程序 的 基础 知识 。 
和 第 1 版 、 修 订 版 一 样 ， 因 为 详细 介绍 了 MySQL 的 基础 知识 ， 所 以 页 数 较 多 。 不 论 是 MySQL， 
还 是 SQL、PHP， 本 书 都 从 基础 开始 进行 了 详细 的 说 明 ， 即 使 是 那些 刚刚 接触 编程 的 人 ， 也 可 以 充 
分 理解 这 些 内 容 。 

在 本 书 第 1 版 诞生 后 的 10 年 间 ，MySQL 的 命运 也 发 生 了 很 大 变化 。 尽 管 如 此 ， 它 依然 是 世界 
上 使 用 人 数 最 多 的 开源 的 RDBMS。 总 之 ,我 们 先 从 MySQL 开始 探索 浩瀚 的 RDBMS 世界 吧 ! 











































































































有 感 于 世界 的 巨大 变化 
2017 年 6 月 30 日 西 泽 梦 路 


本 书 的 阅读 方法 


阅读 方法 

本 书 将 从 基础 知识 开始 介绍 MySQL。 

在 第 1 部 分 至 第 3 部 分 的 内 容 中 ， 我 们 将 通过 命令 提示 符 使 用 基于 CUI 的 “MySQL 监视 器 ” 
来 学 习 MySQL。 其 中 涵盖 了 MySQL 的 概要 和 安装 、SQL 的 基础 内 容 ， 以 及 数据 库 实践 中 必 不 可 
少 的 存储 过 程 和 事务 的 使 用 等 与 MySQL 相关 的 基础 知识 。 

第 4 部 分 至 第 5 部 分 将 介绍 通过 浏览 需 操 作 MySQL 的 方法 。 现 在 对 MySQL 来 说 ，Web 是 不 
可 或 缺 的 ， 许 多 网 站 都 使 用 了 MySQL。 脚 本 语言 PHP 是 从 浏览 器 上 操作 MySQL 最 受 欢迎 的 一 种 
方式 。 我 们 将 学 习 PHP 的 基础 知识 ， 并 不 断 深入 ， 最 终 带领 大 家 使 用 MySQL 和 了 HP 开发 能 够 在 
Web 上 公开 的 安全 性 较 高 的 Web 应 用 程序 。 

虽然 我 尺 自 己 所 能 详细 介绍 了 相关 内 容 ， 但 大 家 仍 有 可 能 遇 到 无 法 理解 或 者 程序 不 能 像 介 绍 的 
那样 顺利 运行 的 情况 。 这 时 最 好 重新 尝试 一 下 ， 待 完全 理解 后 再 继续 学 习 ， 当 然 你 也 可 以 选择 继续 
往 下 阅读 。 本 书 多 次 出 现 “ 一 x x 节 ” 这 种 请 读者 参考 其 他 章节 的 提示 ， 目 的 就 是 让 大 家 通过 参 
考 前 面 的 内 容 来 逐渐 加 深 理 解 。 第 一 次 阅读 时 无 法 理解 的 内 容 ， 到 第 二 次 阅读 时 也 许 就 能 轻松 理 
解 。 另 外 ， 附 录 中 还 准备 了 一 个 “常见 问题 的 检查 清单 ”以 应 对 在 使 用 MySQL 的 过 程 中 可 能 会 发 
生 的 问题 。 希 望 大 家 可 以 坚持 不 懈 地 学 习 下 去 。 





































































































开源 软件 不 好 用 ? 


本 书 用 到 的 MySQL、Apache 和 PHP 都 是 开源 软件 。 如 果 你 只 使 用 过 运行 在 Windows 上 的 应 
程序 ， 那 么 通过 接触 这 些 开源 软件 ， 就 能 发 现 开源 软件 好 的 方面 和 不 好 的 方面 。 

从 安装 到 实际 操作 ， 学 习 MySQL 有 很 长 的 路 要 走 。 此 外 ， 因 为 MySQL 本 来 就 是 在 英语 环境 
下 开发 出 来 的 应 用 程序 ， 所 以 在 使 用 中 文 时 难免 会 遇 到 一 些 问题 。 

在 本 书 中 ,我们 将 使 用 “不 容易 安装 失败 的 MAMP 软件 ”"， 按 照 “ 批 量 安装 Apache + MySQL + 
PHP”“ 使 用 Windows 的 命令 行 操作 数据 库 ” 的 流程 进行 介绍 。 这 在 MySQL 的 专家 看 来 可 能 不 够 
规范 ， 但 是 ， 先 在 容易 使 用 的 环境 中 不 断 进行 尝试 并 解决 出 现 的 问题 也 是 非常 重要 的 。 

在 看 推理 小 说 的 时 候 ， 如 果 从 一 开始 就 知道 谁 是 罪犯 就 没有 意思 了 。 让 我 们 通过 整 本 书 的 交流 
学 习 ， 彻 底 掌 握 Apache + MySQL +PHP 的 技术 吧 ! 
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示例 文件 的 下 载 
下 载 方法 
书 中 介绍 的 开发 环境 、 示 例 程序 ， 以 及 用 于 创建 表 的 脚本 等 ， 都 可 以 通过 以 下 网 址 下 载 。 


示例 文件 的 下 载 
http://Wwww.ituring.com.cn/book/2571 














解压 缩 下 载 的 ZIP 文件 ,会 出 现下 面 这 样 的 文件 夹 。 







































































[chapter ee 各 章 的 PHP 脚 本 文件 和 文本 文件 
to_htdocs) essennes 需要 复制 到 CMAMP\htdocs 的 文件 和 文件 夹 
table: me 利用 SOURCE 命 令 创建 表 时 需要 用 到 的 文本 文件 
——{MAMP] aan MAMP 安 装 程序 
ReadMe.txt 
请 将 上 面 的 文件 夹 复制 到 C 盘 的 根 目录 中 。 
示例 文件 的 使 用 方法 


@ chapter 文件 夹 
chapter 文件 夹 中 按照 章节 保存 了 第 14 章 到 第 21 章 出 现 的 PHP 脚本 文件 和 文本 文件 。 在 学 习 


各 章 的 时 候 请 分 别 参考 相应 的 内 容 。 另 外 ， 包 含 了 中 文 的 PHP 脚本 都 使 用 了 字符 编码 UTF-8。 




















@ to_htdocs 文件 夹 

to_htdocs 文件 夹 统一 保存 了 本 书 中 出 现 的 PHP 脚本 文件 等 内 容 。 如 果 在 MySQL 运行 的 情况 下 
将 这 个 文件 夹 复制 到 你 的 计算 机 上 ， 该 文件 夹 就 可 以 作为 示例 运行 。 

如 果 按 照 本 书 第 2 章 介 绍 的 方法 安装 了 MAMP， 请 将 这 个 to_htdocs 文件 夹 中 包含 的 文件 和 文 
件 夹 复制 到 所 用 计算 机 的 C:\MAMP\htdocs 文件 夹 中 。 在 Apache 和 PHP 等 正常 运行 的 情况 下 ， 如 
果 想 使 用 a.php， 就 可 以 把 http://localhost/a.php 输入 到 浏览 器 的 地 址 栏 里 ， 这 样 页 面 就 会 正常 运行 。 
另外 ,包含 了 中 文 的 示例 都 使 用 了 字符 编码 UTF-8。 





























@ table 文件 夹 
table 文件 夹 中 保存 着 记述 了 SQL 语句 的 文本 文件 ， 这 些 文本 文件 用 于 创建 本 书 中 出 现 的 表 。 
书 中 使 用 的 表 也 就 10 行 左 右 ， 建 议 大 家 手动 输入 ， 权 当 练 习 。 当 然 也 可 以 使 用 table 文件 夹 中 


的 文本 自动 生成 这 些 表 。 























文件 夹 中 包含 以 下 文本 ( 见 表 1 )。 
> 表 1 table 文件 夹 的 内 容 


文件 名 可 以 生成 的 内 容 

































































tb1_make.txt 生成 表 tb1， 并 插入 数据 
tb_make txt 生成 表 tb， 并 插入 数据 
tb2_make .txt 生成 表 tbp2， 并 插入 数据 
tb3_make.txt 生成 表 tb3， 并 插入 数据 
tbj_make txt 生成 表 tbj0 和 tbj1 
tbk_make.txt 生成 表 tbk 
tb1A_K_make.txt 生成 和 表 tb1 内 容 相 同 的 表 tb1A~tb1K， 并 插入 数据 
下 面 这 些 文本 用 于 创建 各 章节 中 使 用 的 表 ， 以 及 插入 说 明 中 需要 的 数据 。 

















t now_ make.txt (一 8.2.6 节 )、 t name maketxt( 一 7.10 节 )、t serial make.txt (— 6.8.2 龙 )、 
t tran make.txt(— 13.6 节 )、t_stock make.txt (一 9.4 节 ) 


例如 当 创 建 表 tbl 时 ,需要 执行 以 下 操作 。 
将 tbl_make.txt 文件 复制 到 C 盘 的 data 文件 夹 中 ,启动 MySQL 监视 器 (一 3.3 节 ) 并 选择 数 
据 库 (use 数据 库 名 称 )， 然 后 执行 下 面 的 命令 。 


mysql> SOURCE C:/data/tbl make.txt 


tbl_make.txt、tb2 make.txt、tb3_make.txt 和 tblA K _make.txt 中 包含 的 中 文 都 使 用 了 字符 编码 
GB 2312。 




















MAMP 文件 夹 


该 文件 夹 中 包含 了 MAMP 的 安装 程序 MAMP_MAMP_PRO 3.3.1.exe。 请 按照 2.2 节 的 说 明 安 
装 MAMP。 
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第 1 部 分 
初 识 MySOL 


MySQL 的 概要 


MySQL 的 环境 配置 





现在 我 们 要 开始 学 习 MySQL 了 。 

在 第 1 部 分 的 内 容 中 ， 我 们 会 在 实际 操作 数据 库 之 前 对 MySQL 和 数据 库 进行 简要 说 明 ， 
并 安装 MAMP。MySOL 的 学 习 在 安装 这 个 环节 就 已 经 开始 了 。 从 安装 到 路 径 设置 ， 再 到 中 
文 的 设置 ， 处 处 都 是 挑战 ， 大 家 要 明白 这 些 都 是 MySQL 的 相关 知识 。 如 果 学 会 使 用 MAMP， 
就 可 以 在 短 时 间 内 轻松 构建 好 Apache + MySQL + PHP 环境 了 。 
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第 1 章 ”MySOL 的 概要 


在 实际 操作 数据 库 之 前 ， 我 们 先 来 了 解 一 下 什么 是 MySQL。 
想 马 上 接触 MySQL 的 读者 也 可 以 跳 过 本 章 ， 直 接 从 第 2 章 
始 阅 读 。 



































1.1 数据 库 的 概要 


1 数据 库 是 什么 

MySQL 是 世界 上 最 受 欢 迎 的 开源 数据 库 软 件 。 那 么 ， 我 们 常常 听 到 的 数据 库 到 底 是 什 
么 呢 ? 

据说 ,第 二 次 世界 大 战 后， 美军 为 了 有 效 管理 大 量 的 资料 ， 便 把 所 有 的 信息 都 集中 在 一 个 基地 
里 ， 这 个 集中 了 所 有 信息 的 基地 就 称 为 数据 库 (database ) “数据 库 ” 一 词 便 由 此 诞生 。 

现在 ， 数 据 库 表示 “具有 某 种 规则 的 数据 集合 "。 但 提 到 数据 库 时 ， 我 们 一 般 都 默认 它 具 备 对 
数据 进行 添加 、 查 询 和 提取 等 用 于 管理 数据 的 功能 。 所 以 ， 只 是 随便 收集 起 来 的 数据 的 集合 不 能 称 
为 数据 库 。 只 有 具备 了 有 效 运 用 这 些 数据 的 管理 功能 ， 才 能 称 为 数据 库 ( 见 图 1-1 )。 
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图 1-1 数据 库 是 什么 


1.1.2 ”关系 数据 库 是 什么 


现在 使 用 最 为 广泛 的 数据 库 是 关系 数据 库 ( Relational DataBase，RDB )。 

在 关系 数据 库 中 ， 一 条 数据 用 多 个 项 目 来 表示 。 例 如 ， 关 系数 据 库 将 一 条 会 员 数 据 分 成 会 员 编 
号 、 姓 名 、 住 址 和 出 生年 月 日 等 项 目 ， 然 后 把 各 个 会 员 的 相关 数据 收集 起 来 。 

其 中 ， 一 条 数据 称 为 记录 (record )， 各 个 项 目 称 为 列 (column )。 在 刚才 的 例子 中 ，x x 先生 
或 者 x x 小 姐 的 数据 是 记录 ， 会 员 编导 和 姓名 等 项 目 是 列 。 

如 果 想 象 成 Excel 的 工作 表 ( work sheet )， 横 向 的 一 行 就 相当 于 记录 。 注 
入 的 是 相同 类 型 的 数据 ( 见 图 1-2 )。 

我 们 把 收集 了 这 些 数据 的 表格 称 为 表 (table )。 一 个 数据 库 中 可 以 包括 多 个 表 。 
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， 纵 向 的 一 列 中 输 






















































表 3 
分 为 多 个 具有 
人 | 工交 1 


表 2 - | ; 
编号 ， 成绩 1 ， 成 绩 2， 成 绩 3 ; … 
表 1 : : : 
编号 ， 姓 名 ' 邮编 ; 住址 ， … 
| 





















































图 1-2 表 、 记 录 和 列 
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管理 关系 数据 库 的 系统 称 为 RDBMS ( Relational DataBase Management System， 关 系数 据 库 管 
理 系 统 )，MySQL 也 是 RDBMS 的 一 种 。RDBMS 是 以 IBM 公司 的 埃 德 加 … 弗兰克: 科 德 (Edgar 
F. Codd ) 于 1970 年 发 表 的 关系 数据 库 相关 论文 为 基础 发 展 起 来 的 。 

在 关系 数据 库 中 ， 不 用 把 所 有 的 项 目 都 存 人 一 个 表 里 。 我 们 可 以 把 各 个 项 目 拆 到 多 个 “具有 关 
联 性 ”的 表 中 ， 只 对 需要 用 到 的 数据 进行 收集 和 使 用 。 























5 1.1.3 数据 库 的 特征 


如 果 是 Excel 的 工作 表 ， 在 任何 单元 格 中 都 可 以 自由 地 输入 字符 串 或 数值 ， 而 且 还 可 以 通过 拖 
忠 鼠 标 来 自由 地 挪动 数据 。 

但 是 在 数据 库 中 ， 上 面 的 做 法 是 行 不 通 的 。 如 果 最 开始 决定 “在 列 a 中 输入 整数 ”"， 之 后 在 列 a 
中 就 只 能 输入 整数 ， 不 能 再 输入 字符 串 。( 严格 来 讲 ， 在 MySQL 中 即使 输入 了 不 同类 型 的 数据 也 不 
会 报错 ， 但 是 结果 可 能 变 成 什么 值 都 没有 搬入 进去 ， 这 一 点 需要 特别 注意 。 一 5.1 节 ) 

另外 ， 很 多 应 用 程序 有 还 原 (UNDO ) 功能 ， 如 果 操 作 结 果 不 理想 ， 可 以 使 用 该 功能 进行 还 
原 。 但 是 在 数据 库 中 ， 如 果 不 使 用 事务 (transaction ) (一 13.3 节 ) 等 特殊 功能 ， 就 无 法 还 原 到 上 一 


步 操作 〈 见 图 1-3 )。 
可 
表 还 原 操作 > 
INT 类 型 ' …  ' VARCHAR(10) 类 型 ， 一 … 

























































































5 字符 1 

处 理 大 量 数 据 a 字符 2 
8 字符 3 
字符 4 
9 字符 5 
3 


字符 6 











图 1-3 数据库 的 特征 
大 家 也 许 一 开始 会 讨厌 数据 库 这 种 “缺乏 灵活 性 ”的 特性 。 但 多 亏 了 这 一 特性 ， 数 据 库 才 变 得 
安全 且 稳 定 ， 才 成 了 可 靠 的 数据 存储 场所 ( 见 图 1-4 ) “只 能 按照 决定 好 的 规则 来 操作 ， 并 且 严 格 
地 进行 管理 ”的 特征 ， 正 是 数据 库 值得 信赖 的 原因 。 


二 大 量 的 
回 图 


1-4 数据 库 的 特征 





























1.2 MySQL 是 什么 样 的 数据 库 | 5 


1.2 MySOQL 是 什么 样 的 数据 库 


下 面 将 介绍 MySQL 和 其 他 RDBMS 有 哪些 不 同 ， 以 及 开源 软件 究竟 为 何 物 。 





局 12.1 MySQL 是 开源 软件 


MySQL 是 一 种 RDBMS (一 1.1.2 节 )。 现 在 世界 上 有 许多 种 RDBMS 被 广泛 使 用 。 表 1-1 中 列 
出 了 一 些 主流 的 数据 库 。 








表 1-1 主流 数据 库 













































































名 称 特征 

Oracle 世界 上 最 常用 的 商用 RDBMS 

Access 微软 公司 Office 系列 的 RDBMS 

Microsoft SOL Server 微软 公司 的 商用 RDBMS 

PostgreSOL 和 MySQL 一 样 是 开源 的 RDBMS， 在 日 本 很 受 欢迎 
MySOL 世界 上 最 常用 的 开源 RDBMS 





























Oracle 和 Microsoft SQL Server 是 商用 数据 库 。 使 用 这 些 数据 库 时 需要 付费 ， 还 要 签订 授权 许 
可 协议 。MySQL 和 PostgreSQL 是 开源 数据 库 。 这 些 开 源 的 应 用 程序 可 以 免费 使 用 ， 并 允许 他 人 对 
其 进行 修改 。 只 要 通过 互联 网 下 载 相关 程序 就 可 以 自由 地 使 用 了 。 

但 这 并 不 表示 开源 应 用 程序 可 以 被 随意 使 用 ， 开 源 应 用 程序 的 使 用 也 有 一 些 规 则 和 限制 。 特 别 
是 在 商用 的 情况 下 ， 使 用 前 一 定 要 充分 理解 和 掌握 使 用 协议 的 内 容 。 

PostgreSQL 虽然 作为 开源 RDBMS 在 日 本 非常 受 欢 迎 ， 但 从 世界 范围 来 看 ， 还 是 MySQL 更 胜 
一 筹 。 目 前 世界 上 有 超过 1000 万 台 服 务 器 安装 并 使 用 MySQL， 可 以 说 MySQL 是 世界 上 最 常用 的 
开源 RDBMS。 在 互联 网 领域 , 深 受 欢迎 的 Yahoo ! 、Twitter、YouTube 和 Facebook 等 网 站 的 数据 
库 现 在 也 都 使 用 了 MySQL。 需 要 提 一 下 的 是 ，Google 正 从 MySQL 向 MariaDB 进行 切换 。( 2017 
年 6 月 时 的 信息 ) 







































































S 1.2.2 MySQL 的 历史 


MySQL 是 米 卡 埃 尔 。 维 德 纽 斯 ( Michael Widenius ) 在 1995 年 开发 的 RDBMS。 最 初 ，MySQL 
是 由 瑞典 的 MySQL AB 公司 进行 支持 和 开发 的 。 维 德 纽 斯 正 是 这 家 公司 的 前 身 公 司 的 创始 人 。 但 
在 2008 年 2 月 ，MySQL AB 公司 被 Sun 公司 收购 。 而 拥有 被 认为 是 世界 第 一 的 商用 数据 库 Oracle 
的 Oracle 公司 又 于 2010 年 1 月 收购 了 Sun 公司。 也 就 是 说 ， 作 为 开源 数据 库 世 界 第 一 的 MySQL 
和 作为 商用 数据 库 世 界 第 一 的 Oracle， 现 在 都 由 同一 家 公司 管理 。 虽 然 全 世界 都 在 关注 Oracle 公司 
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的 下 一 步 举措 ， 但 目前 为 止 Oracle 和 MySQL 依然 在 各 自 擅长 的 领域 作为 优秀 的 RDBMS 良好 地 发 
展 着 。 

另外 ， 维 德 纽 斯 原 供职 于 Sun 公司 ,但 是 他 在 Oracle 公司 收购 Sun 公司 前 从 Sun 公司 离职 ， 
并 于 2010 年 创建 了 MariaDB 公司 ， 开 始 开 发 和 MySQL 具有 兼容 性 的 MariaDB 数据 库 。MySQL 
源 于 维 德 纽 斯 大 女儿 的 名 字 “My”， 而 MariaDB 则 源 于 维 德 纽 斯 二 女儿 的 名 字 “Maria”。 

MariaDB 最 初 发 布 的 版 本 只 是 增强 了 MySQL 的 部 分 功能 ， 但 是 从 2014 年 3 月 发 布 的 10.0 版 
本 起 ，MariaDB 开始 有 了 自己 专 有 的 功能 ， 逐 渐 和 MySQL 区 分 开 来 。 不 过 在 本 书 介绍 的 内 容 中 ， 
MySQL 和 MariaDB 的 基本 操作 方法 并 没有 什么 区 别 。 



































MySQL 主要 有 以 下 两 种 版 本 。 


PP MySQL 社区 版 ( Community Edition ) 
2 可 免费 使 用 
2 有 参考 手册 (reference manual ) 
2 有 论坛 (forum ) 和 邮件 列表 ( mailing list )， 但 没有 技术 支持 





区 MySQL 商业 版 (Commercial Edition ) 


2 需要 付费 
2 定期 更 新 ， 并 提供 服务 包 〈 service pack ) 及 技术 支持 
2 拥有 以 下 多 个 版 本 

MySQL 标准 版 ( Standard Edition ) 

MySQL 企业 版 ( Enterprise Edition ) 

MySQL 集群 运营 商 级 版 ( Cluster Carrier Grade Edition ) 























本 书 介绍 的 是 可 以 免费 使 用 的 MySQL 社区 版 。MySQL 社区 版 重视 加 入 新 功能 ， 在 一 定 的 限 
制 条 件 下 可 以 自由 使 用 。 

而 MySQL 商业 版 面向 的 是 那些 需要 售后 技术 支持 的 企业 。 商 业 版 更 注重 稳定 性 ， 可 以 保证 运 
营 和 管理 按照 计划 进行 。MySQL 商业 版 面向 企业 等 客户 ,为 了 实现 高 可 用 性 和 高 效率 的 运营 ， 提 
供 了 丰富 的 软件 和 全 面 的 技术 支持 服务 。 




































































5 1.2.4 ” MySQL 的 特征 


MySQL 有 以 下 几 个 特征 。 
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1.2 ”MySQL 是 什么 样 的 数据 库 


1. 执行 速度 快 

2. 开放 源 代码 

3. 支持 在 多 种 操作 系统 上 运行 
4. 支持 多 种 编程 语言 

5. 拥有 免费 和 付费 两 种 版 本 








MySQL 的 执行 速度 非常 快 ， 这 种 轻快 性 正 是 它 的 卖点 。 早 期 的 版 本 为 了 维持 这 种 轻快 性 ， 
削减 事务 、 子 查询 (一 10.5 节 ) 和 存储 过 程 (一 12.1 节 ) 等 功能 ， 所 以 给 人 一 种 “执行 速度 快 ， 
但 功能 太 简 单 ” 的 感觉 。 不 过 ， 现 在 MySQL 在 功能 方面 已 经 不 逊色 于 其 他 RDBMS 了 。 

表 1-2 是 MySQL 新 功能 的 添加 历史 。 








表 1-2 MySQL 新 功能 的 添加 历史 


添加 该 功能 时 的 MySQL 版 本 























































































































事务 (一 13.3 节 ) 4.0 ( 从 版 本 3.23.38 开始 可 以 使 用 该 功能 ) 
合 >10.1 节 ) 40 

子 查询 ( 一 10.5 节 41 

视图 (一 11.1 节 ) 5.0 
存储 过 程 (一 12.1 节 ) 5.0 
存储 函数 (一 12.4 节 ) 5.0 
触发 器 ( 一 12.6 节 5.0 
默认 存储 引擎 变 为 InnoDB (一 13.1.2 节 ) 5.5 

半 同 步 复制 5.5 

无 损 半 同步 复制 5.7 

※ 编写 本 书 时 ( 2017 年 6 月 )，MySQL 的 最 新 稳定 版 本 是 5.7.17 






































虽然 MySQL 的 许多 竞争 对 手 从 一 开始 就 具备 了 这 些 功能 ， 但 是 现在 MySQL 也 具备 了 与 其 他 
RDBMS 相 媲 美的 功能 。 

如 果 问 使 用 过 数据 库 的 人 哪个 RDBMS 最 好 ， 每 个 人 都 会 自信 满 满 地 推荐 自己 喜欢 的 数据 库 。 

到 底 哪个 RDBMS 最 好 ? 这 个 问题 并 没有 一 个 标准 的 答案 。 但 是 ，MySQL 是 世界 上 最 常用 的 
开源 RDBMS 却 是 一 个 不 争 的 事实 。 能 够 成 为 世界 上 最 受 欢 迎 的 数据 库 必 然 有 它 的 理由 。 


















































(D 本 书 翻译 时 (2018 年 3 月 ) MySQL 的 最 新 稳定 版 本 是 5.7.21， 最 新 的 社区 开发 版 本 是 8.0.4。 译 者 注 
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1.3 SQL 的 概要 


S 1.3.1 什么 是 SQL 


P 查询 和 SQL 


在 操作 数据 库 的 时 候 ， 作 为 用 户 的 我 们 会 向 数据 库 发 出 命令 (command )， 并 指定 需要 处 理 的 内 
容 。 表 示 这 种 命令 的 语句 就 是 查询 (query )。 例 如 ， 创 建 表 时 使 用 的 查询 是 “CRERATE TABLE ...”， 
插入 数据 时 使 用 的 查询 是 “INSERT INTO...”。 

编写 查询 需要 遵守 SQL ( Structured Query Language ) 的 规则 。SQL 直译 过 来 就 是 结构 化 查询 
语言 ， 用 于 对 数据 库 进行 操作 。 

我 们 来 看 一 个 例子 。 假 设 你 在 政府 办 公 室 等 地 方 进 行 咨询 ， 在 向 政府 办 公 室 窗口 的 负责 人 提出 
“ 想 请 x x 部 门 帮忙 提供 x x 证 明 ” 的 申请 后 ， 就 能 得 到 需要 的 文件 了 。 如 果 把 申请 对 象 换 成 数 
据 库 ， 申 请 手续 就 是 SQL。SQL 是 数据 库 的 窗口 ， 充 当 用 户 与 数据 库 交互 的 媒介 。 

实际 上 ， 即 使 不 使 用 SQL 这 种 语言 向 数据 库 发 送 命令 ， 也 可 以 使 用 一 些 工 具 通 过 直观 的 鼠标 
操作 来 编辑 和 显示 数据 。 例 如 附录 中 介绍 的 phpMyAdmin 就 是 这 样 的 一 个 工具 。 使 用 phpMyAdmin 
就 可 以 不 用 输入 “SELECT * FROM tb ...WHERE ...” 等 复杂 的 SQL 语句 了 。 因 此 ， 也 有 很 多 书 不 
介绍 SQL 语句 的 详细 内 容 ， 而 是 直接 介绍 如 何 利 用 phpMyAdmin 等 工具 来 创建 数据 库 。 

但 是 ， 当 使 用 PHP 等 语言 编写 Web 应 用 程序 时 ， 如 果 需 要 实际 操作 某 个 数据 库 ， 就 必须 输入 
正确 的 SQL 语句 。 所 以 在 本 书 中 ， 我 们 会 一 个 一 个 地 输入 SQL 语句 并 执行 ， 然 后 确认 其 结果 ， 在 
此 过 程 中 学 习 MySQL。 

























































































































































































> SQL 的 “方言 ” 


SQL 原本 是 IBM 公司 开发 的 语言 ， 但 现在 这 门 语言 基本 可 以 在 所 有 的 数据 库 上 使 用 。 不 过 麻 
烦 的 是 ， 各 个 数据 库 使 用 的 SQL 的 语法 稍 有 不 同 。 笔 者 经 常 使 用 PostgreSQL， 但 因 PostgreSQL 和 
MySQL 的 SQL 命令 有 一 些微 妙 的 差异 ， 所 以 也 常常 感到 困惑 。 因 此 ， 如 果 你 已 经 习惯 了 MySQL 
以 外 的 RDBMS ， 就 要 注意 这 种 SQL 的 “方言 ”了 。 




















5 1.3.2 ”首先 熟悉 SELECT 命令 

SQL 中 有 许多 命令 。 其 中 ，SELECT 是 用 于 选择 数据 的 命令 ， 它 也 是 SQL 中 使 用 得 最 为 频繁 
的 关键 字 。 无 论 怎 样 使 用 SELECT 命令 ， 都 不 必 担 心 会 损坏 或 更 改 数据 ， 所 以 在 数据 库 的 入 门 阶 
段 ， 要 尽 可 能 多 地 使 用 SELECT 来 熟悉 SQL。 
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在 附录 3 中 ， 笔 者 准备 了 一 系列 用 于 熟悉 数据 库 操作 的 MySQL 基础 练习 。 想 尽早 自由 使 用 
MySQL 的 读者 ， 请 试 着 每 天 反复 做 这 个 基础 练习 。 相 信 经 过 反复 的 练习 ， 你 会 灵活 使 用 最 初 连 意 
思 都 不 明白 的 SQL。 

成 为 MySQL 专家 的 第 一 步 ， 就 是 和 MySQL 成 为 朋友 。 


本 章 主要 介绍 了 以 下 内 容 。 











@ 什 么 是 数据 库 ， 什 么 是 RDBMS 
@ MySQL 的 特征 

@ 使 用 MySQL 的 两 种 版 本 

@ 数 据 库 的 操作 方法 











对 于 世界 上 最 常用 的 开源 数据 库 MySQL， 大 家 是 不 是 已 经 有 了 大 致 的 了 解 9 下 一 章 我 们 将 会 
安装 MAMP ， 并 进行 MySQL 的 环境 配置 。 


> 自我 检查 
我 们 来 检查 一 下 本 章 学 习 的 内 容 是 否 全 部 理解 并 掌握 了 。 
口 理解 了 数据 库 是 什么 
口 理解 了 MySQL 的 特征 


口 理解 了 查询 和 SQL 是 什么 


第 2 章 ”MySQL 的 环境 配置 








本 章 我 们 将 设置 本 书 中 用 于 学 习 MySQL 的 环境 。 


2.1 本 书 中 使 用 的 软件 


这 里 我 们 来 安装 本 书 中 使 用 的 软件 ， 并 设置 可 以 使 用 MySQL 和 Apache 的 环境 。 
本 书 中 使 用 的 软件 主要 包括 以 下 几 个 。 





























> MySQL (RDBMS ) 
2 Apache ( Web 服务 器 ) 
3 PHP ( 编程 语言 ) 





本 来 这 些 软件 是 需要 分 别 进行 安装 和 配置 的 ， 但 是 分 别 安装 会 花费 很 多 时 间 和 精力 ， 而 且 在 某 
些 使 用 环境 下 还 需要 进行 复杂 的 配置 。 

所 以 建议 大 家 使 用 能 够 轻松 构建 MySQL+Apache+PHP 运行 环境 的 MAMP。 使 用 MAMP 可 以 
非常 简单 并 且 准 确 地 构建 MySQL+Apache+PHP 环境 。MAMP 的 名 字 取 自 Macintosh、Apache、 
MySQL 和 PHP 这 几 个 词 的 首 字 母 。MAMP 中 的 M 是 Macintosh 的 意思 ， 所 以 可 能 有 人 担心 这 个 软 
件 仅 能 在 macOS 上 使 用 ， 但 其 实 它 也 可 以 在 Windows 上 使 用 。 本 书 介绍 的 操作 方法 基本 以 
Windows 环境 为 前 提 。 

在 从 本 书 的 支持 网 站 上 下 载 的 文件 中 ， 也 包含 了 用 于 Windows 环境 的 免费 版 MAMP 的 安装 文 
件 (MAMP MAMP PRO 3.3.1.exe ) "。 请 参考 本 书 前 言 “ 示 例文 件 的 下 载 ” 部 分 下 载 开发 环境 和 示 























































































































QD 截至 2018 年 3 月 ，Windows 版 的 MAMP 的 最 新 版 本 为 MAMP & MAMP PRO 4.0。 





译 者 注 
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例 程 序 ， 具 体 的 安装 方法 将 在 以 后 的 章节 中 介绍 。 

















此 外 ,通过 MAMP 构建 的 环境 主要 以 开发 为 目的 ， 并 没有 达到 能 够 在 互联 网 上 实际 发 布 的 安 


























全 级 别 。 所 以 如 果 要 实际 发 布 到 互联 网 上 ， 还 需要 事先 做 好 充足 的 准备 。 


XAMPP 


XAMPP 和 MAMP 一 样 ， 是 一 个 可 以 轻松 构建 MySOL+Apache+PHP 运行 环境 的 软件 。 详 细 内 容 可 以 参考 








下 面 Apache Friends 的 Web 页 面 。 

















个 XAMPP 
https://www.apachefriends.org/zh_cn/index.html 























本 书 原版 第 1 版 和 修订 版 都 使 用 了 XAMPP 来 构建 MySOL+Apache+PHP 运行 环境 ， 但 是 XAMPP 从 版 本 
5.6.14 开始 使 用 的 数据 库 从 MySQL 换 成 了 MariaDB。 截 至 2017 年 8 月 ，XAMPP 的 最 新 版 本 XAMPP 7.1.8 的 





























数据 库 依然 使 用 了 MariaDB "。 


2.2 学习 MySQL 前 的 准备 ( 安装 和 配置 ) 


下 面 我 们 开始 使 用 MAMP 安装 MySQL。 





S 2.2.1 关于 本 书 使 用 的 MAMP 


如 上 所 述 ， 使 用 MAMP 可 以 轻松 地 构建 MySQL+Apache+PHP 的 运行 环境 。 本 书 使 用 的 
MAMP 3.3.1 的 系统 要 求 和 所 需要 的 主要 软件 一 览 可 参考 表 2-1。 另 外 ，MAMP 包括 可 以 免费 使 用 














的 标准 版 和 需要 付费 的 专业 版 (MAMP PRO ) 两 种 版 本 ， 本 书 使 用 的 是 标准 版 。 


表 2-1 MAMP 3.3.1 的 数据 表 











操作 系统 Windows 10、Windows 8.1 或 Windows 7 





内 存 : 最 低 2 GB 





HDD: 2 GB 的 空闲 空间 





Apache : 2.2.31 





MySQOL : 5.6.34 





PHP: 7.1.5 和 7.0.19 








phpMyAdmin : 4.4.15.5 











四 截至 2018 年 3 月 ，Windows 版 的 XAMPP 的 最 新 版 本 XAMPP 7.2.3 的 数据 库 依 然 使 用 了 MariaDB。 





译 者 注 
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在 MAMP 的 安装 过 程 中 ,除了 上 面 的 软件 之 外 ， 还 有 许多 软件 会 被 安装 ,但 是 本 书 介绍 的 内 
容 并 不 涉及 这 些 软件 。 男 外 ， 虽然 PHP 的 版 本 5 系列 也 被 安装 了 ,但 由 于 在 标准 版 中 仅 能 选择 版 
本 7 系列， 所 以 本 书 使 用 了 PHP 7.1.5。 

此 外 ， 关 于 phpMyAdmin， 附 录 中 介绍 了 简单 的 使 用 方法 。 

MAMP 的 最 新 版 可 以 通过 下 面 的 URL 下载。 但 是 要 注意 ， 如 果 版 本 不 同 ， 具体 的 操作 内 容 就 
可 能 会 和 本 书 介绍 的 不 同 。 




































































© MAMP 
https://www.mamp.info/en/ 


如 果 文件 的 扩展 名 未 显示 

本 书 介绍 的 内 容 以 扩展 名 ( 主 文件 名 后 面 附属 的 “.txt” 等 后 缀 ) 呈 显 示 状 态 为 前 提 。 但 是 ，Windows 环 
境 默认 不 显示 扩展 名 。 所 以 在 没有 显示 扩展 名 的 情况 下 ， 要 事先 把 扩展 名 设置 成 显示 状态 。 在 Windows 10 中 ， 
我 们 只 需 在 文件 资源 管理 器 ( explorer ) 中 将 “查看 ”菜单 中 的 文件 扩展 名 勾 选 上 即 可 。 








































































































福 | 回 文件 资源 管理 器 勾 选 此 处 = 可 xX 
人 人 四 
预 史 让 属 。 [本 超 大 图 标 本 大 图 标 “三 中 国标 ~ 国 | 同 今 8 信 得 口 项 目 复 选 醛 | 》 
Oo 天 小 图 标 “ 肤 列表 肚 详 细 信息 - | 徊 添加 列 ~ 回 文件 扩展 名 | 四 
导航 窗 格 [详细 信息 窗 格 区 有 = = | 排序 方式 隐藏 选项 
四 让 = 内 容 ” 。” 淹 将 所 有 列 洞 整 为 合适 的 大 小 口 隐藏 的 项 目 所 选项 目 
窗 格 布局 当前 视图 显示 /隐藏 


C 2.2.2 ”安装 MAMP 


下 面 介绍 Windows 版 的 MAMP 3.3.1 的 安装 方法 。 书 中 按照 选择 默认 的 路 径 安装 文件 夹 的 方式 
进行 介绍 。 所 以 ， 如 果 按 照 本 书 介绍 的 方法 操作 ， 软 件 会 安装 到 路 径 CNMAMP 中 。 

如 果 仅 仅 安装 了 MAMP， 而 MySQL 和 Apache 都 不 能 正常 使 用 的 话 就 没有 意义 了 。 因 此 ， 大 
家 需要 小 心 谨慎 地 进行 操作 。 

















区 如 果 80 号 端口 和 3306 号 端口 被 占用 就 会 运行 失败 | 

MAMP 会 把 80 号 端口 分 配给 Apache， 把 3306 号 端口 分 配给 MySQL 使 用 。 所 以 ， 如 果 这 些 
端口 已 经 被 占用 ，Apache 和 MySQL 的 相关 服务 就 无 法 运行 。 因 此 ， 如 果 有 占用 这 些 端口 的 应 用 程 
序 或 者 服务 ， 请 禁用 后 再 安装 MAMP。 





























P ”如 果 旧 版 本 的 Apache 和 MySQL 有 残留 也 会 运行 失败 ! 

如 果 以 前 安装 过 Apache 和 MySQL， 并 且 相 关内 容 没有 仓 载 干净 ， 就 可 能 会 出 现 安装 失败 或 者 
无 法 正常 运行 等 情况 。 所 以 ， 请 务必 把 旧 的 Apache 和 MySQL 弛 载 干净 ， 然 后 重新 启动 操作 系统 ， 
安装 MAMP。 
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区 安装 方法 

Q 解压 缩 下 载 文件 ， 执 行 MySQL_Book 文件 夹 的 子 文件 夹 MAMP 中 的 MAMP_MAMP 
PRO 3.3.1.exe 文件 。 

@ 如 果 “ 用 户 账户 控制 ”对 话 框 中 显示 “你 要 允许 此 应 用 对 你 的 设备 进行 更 改 吗 ”"， 选 择 
“是 《Ya 

@) 安装 向 导 (setup wizard ) 启动 后 ， 会 显示 图 2-1 的 画面 ， 点 击 “Next >”。 














© Setup - MAMP & MAMP PRO 六 x 


Welcome to the MAMP & MAMP 
PRO Setup Wizard 

This will install MAMP & MAMP PRO version 3.3. 1 on your 
computer, 

Itis recommended that you dose all other applications before 
continuing. 


Click Next to continue, or Cancel to exit Setup. 








| Next> | | Canal 








图 2-1 启动 安装 向 导 
由 在 图 2-2 的 画面 中 ， 取 消 “MAMP PRO” 的 选中 状态 ， 点 击 “Next >”。 
































© Setup - MAMP & MAMP PRO 一 Xx 
MAMP and MAMP PRO 
Please provide additional setup options. 
Install MAMP PRO in addition to MAMP. 
口 MAMP PRO 取消 选中 
| <Bak | Net> | | Cane 











2-2 MAMP and MAMP Pro 的 画面 
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@) 图 2-3 为 是 否 同 意 授 权 许 可 证 的 确认 画面 ， 请 选择 “I accept the agreement”， 然 后 点 击 


“Next Ss” 



































© Setup - MAMP & MAMP PRO 一 X 


License Agreement 
Please read the following important information before you continue. | 





Please read the following License Agreement, You must accept the terms of this 
agreement before continuing with the installation 





TU ME SO CUM MU Me Cen sm 


Iprovide these to you on a purely Eoodwill basis. The Licensor has no influence 


whatsoever over the content of these websites and bears no responsibility with 
regard hereto. The provider in question shall always bear any and all responsibility. 


?73 In the event that individual provisions of the present licensing agreement should 
be declared invalid ilegal null and void or unenforceable, the remaining provisions shall 


continue to retain their validity. 


Last amended: O01 06 .2011 v 


@®) I accept the agreement 
OI do not accept the agreement 


























图 2-3 同意 授权 许可 证 
@ 在 图 2-4 的 画面 中 ， 保 留 “CNMAMP”， 点 击 “Next >”。 





© Setup - MAMP & MAMP PRO 
Select Destination Location 术 二 
Where should MAMP & MAMP PRO be installed? | 有 


Setup will install MAMP & MAMP PRO into the following folder. 





Click Next to continue. Click Browse to select a different folder. 


C:\MAMP Browse... 

















At least 1,311.1 MB of free disk space is required. 











< Back Next > Cancel 





























2-4 ”指定 安装 路 径 
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@ 在 图 2-5 的 画面 中 ， 指 定 在 开始 菜单 中 显示 的 文件 夹 名 。 因 为 没有 必要 进行 修改 ， 所 以 保留 
“MAMP”， 点 击 “Next >”。 








© Setup - MAMP & MAMP PRO ny x 
Select Start Menu Folder 1 
Where should Setup locate the program's shortcuts? | | 





了 
Setup will locate the program's shortcuts in the folowing Start Menu folder. 


Click Next to continue. Click Browse if you want to select a different folder, 





IMAMP Browse... 












































图 2-5 指定 创建 快捷 方式 的 文件 夹 
@@ 为 了 在 桌面 上 创建 MAMP 的 快捷 方式 ， 保 持 选择 框 的 选中 状态 ， 点 击 “Next >”( 见 图 2-6 )。 




















© Setup - MAMP & MAMP PRO = x 
Select Additional Tasks b 
Which additional tasks should be performed? | | 





Select all additional tasks you would like Setup to perform while instaling MAMP & 
MAMP PRO, then dick Next. 


















































图 2-6 指定 在 桌面 上 创建 MAMP 的 快捷 方式 
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@@ 图 2-7 的 画面 会 显示 要 安装 的 内 容 ， 确 认 后 点 击 “Install”。 








© Setup - MAMP & MAMP PRO 3 


Ready to Install 李 时 
Setup is now ready to start installing MAMP & MAMP PRO on your computer. 


Click Install to continue with the installation, or click Back if you want to review or 
change any settings. 








Destination location: 
C:\MAMP 


Start Menu folder: 
MAMP 


Additional tasks: 
Additional icons: 
Create a desktop icon 


























< Back Install Cancel 














图 2-7 确认 安装 内 容 
0 这 样 安装 就 开始 了 。 安 装 结束 后 ， 会 显示 图 2-8 的 画面 ， 点 击 “Finish”，MAMP 就 安装 完 
成 了 。 














© Setup - MAMP & MAMP PRO = 


Completing the MAMP & MAMP 
PRO Setup Wizard 


Setup has completed the installation of MAMP & MAMP PRO on 
your computer. The application may be launched by selecting 
the installed icons. 


Click Finish to exit Setup. 











2-8 安装 向 导 结束 


洲 


2.2.3 MAMP 的 启动 和 设置 





安装 完成 后 ， 就 可 以 立即 尝试 启动 MAMP 了 。 点 击 桌 面 上 的 MAMP 图 标 ， 或 者 选择 “开始 ” 
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菜单 的 “MAMP” 一 “MAMP”， 就 可 以 启动 MAMP 了 。 
启动 MAMP 后 会 显示 图 2-9 的 画面 。 在 初始 设置 的 情况 下 ， 
也 会 自动 启动 。 





启动 MAMP 时 Apache 和 MySQL 





© MAMP 一 


A MAMP 


manage your websites locally 


女 


Preferences... Open start page Stop Servers 













Apache 和 MySQL 运 
行 时 会 显示 为 绿色 




































































点 击 这 里 停止 Apache 
























































图 2-9 MAMP 的 启动 画面 





我 们 可 以 通过 这 个 画面 控制 Apache 和 MySQL。 如 果 想 停止 Apache 和 MySQL， 则 点 击 “Stop 
Servers”。 在 停止 的 情况 下 ， 按 钮 会 变 成 “Start Servers”， 再 次 点 击 就 可 以 启动 Apache 和 MySQL 了 。 
当 对 MAMP 进行 各 种 设置 时 ， 点 击 “Preferences” 会 显示 出 图 2-10 的 画面 。 我 们 可 以 通过 切 
换 画 面 上 方 的 标签 来 设置 相关 内 容 。 本 书 没有 什么 项 目 需要 更 改 设 置 ， 所 以 这 里 仅 介绍 设置 方法 。 
在 “Start/Stop” 标 签 ( 见 图 2-10 ) 中 ， 我 们 可 以 设置 MAMP 启动 时 各 服务 器 的 相关 动作 。 




















Preferences 











Start/Stop | Ports | PHP | web Server | About MAMP 











Start Servers when starting MAMP ©@—— 





选中 则 设置 为 启动 MAMP 的 
同时 也 启动 Apache 和 MySQL 











[DD Stop Servers when quitting MAMP 
[DD Check for MAMP PRO when starting MAMP 
口 ] At startup open 


Start page url 











选中 则 设置 为 停止 MAMP 的 
同时 也 停止 Apache 和 MySQL 











/MAMP/ 





























2-10 “Start/Stop” 标 签 
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在 “Ports” 标 签 ( 见 图 2-11 ) 中 ， 我 们 可 以 设置 Web 服务 器 和 MySQL 服务 需 使 用 的 端口 号 。 
如 果 80 号 端口 和 3306 号 端口 被 其 他 程序 占用 了 ， 也 可 以 设置 成 别 的 端口 号 以 便 和 其 他 程序 共存 。 
另外 ，Nginx 是 和 Apache 一 样 用 于 构建 Web 服务 器 的 软件 ， 但 本 书 没有 使 用 它 。 














Preferences 回 




















Start/Stop | Ports | PHP | Web Server | About MAMP 











Apache Port: 80 (1-65536) 














Nginx port: 80 (1-65536) Apache 和 MySQL 使 用 的 端 






































MySQL Port: 3306 (1-65536) 
Set MAMP ports to default 
Set Web & MySQL ports to 80 & 3306 























2-11 “Ports” 标 签 


在 “PHP” 标 签 ( 见 图 2-12 ) 中 ， 我 们 可 以 设置 使 用 的 PHP 版 本 以 及 缓存 相关 的 功能 。 在 
MAMP 3.3.1 的 情况 下 ， 可 以 选择 PHP 7.1.5 或 PHP 7.0.19。 本 书 使 用 了 默认 的 7.1.5 版 本 。 


FI 


虽然 PHP 的 缓存 功能 包括 Apcu 和 Zend OPcache 两 种 类 型 ， 但 这 里 保留 默认 的 关闭 缓存 










































































Preferences 图 
Start/Stop | ports | PHP | Web Server | About MAMP 
选择 PHP 的 版 本 。 本 书 
Standard Version: | 7.1.5 使 用 的 是 7.1.5 
Cache: | off ~ 选择 缓存 功能 的 类 型 











2-12 “PHP” 标 签 
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在 “Web Server” 标 签 ( 见 图 2-13 ) 中 ， 我 们 可 以 设置 Web 服务 器 的 种 类 和 文件 根 目 录 。 文 件 
根 目录 (document root ) 是 Web 服务 器 放置 发 布 文件 的 最 上 一 层 目录 。 默 认 目 录 是 CAMAMP' 
htdocs， 可 以 通过 http:Wlocalhost 访问 。 








Preferences 








| Start/Stop | Ports | PHP | Web Server | About MAMP | 




















Webserver ”图 Apache 〇 Nginx @ 一 一 一 一 一 在 这 里 选择 Web 服务 器 





Document Root: 




















件 根 目 录 的 文件 夹 。 书 中 使 
默认 目录 CMAMP\htdocs 
| Select... | | 


Open 


























[CNMAMPNhtdocse， | 















































[ ok | [ Cancel | 








图 2-13 “Web Server” 标 签 


什么 是 localhost 

localhost 就 是 本 地 主机 ， 即 当前 使 用 的 计算 机 的 名 称 ( 主机 名 )。 

localhost 对 应 的 IP 地 址 为 127.0.0.1 ( IPv4 ) 和 ::1 ( IPv6 )。 这 是 用 于 表示 主机 自身 的 特殊 IP 地 址 ， 称 为 
本 地 回环 地 址 ( local loopback address)。 这 个 localhost 可 以 跟 主 机 自身 进行 通信 测试 。 






































































































































C 2.2.4 ”确认 开始 页 


下 面试 着 打开 MAMP 的 开始 页 。 我 们 可 以 通过 开始 页 确认 PHP 和 MySQL 的 设置 内 容 。 在 
MAMP 的 启动 画面 上 ， 点 击 “Open start page”( 见 图 2-14 )。 
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个 七 htpy/localhosbyMAMP pro |@ vawp 








Welcome! 


MAMP has been 
installed 
successfully. 点 旦 显示 PHP 的 各 











PHP 


phpinfo shows the current configuration of PHP. 


MySQL 


MySQL can be administered with phpMyAdmin 


To connect to the MySQL server from your Own scripts Use the following connection parameters; 














Host localhost MySOQL 服务 器 的 设置 情况 
Port 3306 
User root 


Password root 


Socket JApplications/MAMPtmp/mysqymysql.sock 


Examples 


PHP <= 5.5X PHP >= 56X Python Perl 


2-14 MAMP 的 开始 页 
从 MySQL 服务 器 的 设置 概况 中 可 以 看 到 ， 用 户 名 和 密码 都 是 “root”。 男 外 ， 点 击 画 面 上 方 的 
“Tools”， 选 择 菜 单 中 的 “phpMyAdmin”， 就 可 以 启动 bbppMyAdmin 了 。 


























2.2.5 构成 MAMP 的 文件 夹 


如 果 将 MAMP 安装 在 路 径 CN\MAMP 中 ， 构 成 它 的 文件 夹 就 是 下 面 这 样 。 












































除 此 之 外 还 有 一 些 文件 夹 ， 但 本 书 并 不 会 使 用 ， 所 以 这 里 省 略 了 相关 介绍 。 





2.3 MySQL 的 路 径 设置 














Apache 公 


的 文件 夹 


et ee dn Apache 服 务 器 的 文件 夹 


a MySQL 服 务 器 的 文件 夹 


Sn PHP 的 文件 夹 
ooooos0osecosooso0s 保存 各 种 设置 文件 的 文件 夹 


A 保存 MySQL 数 据 的 文件 夹 


2.3 MySQL 的 路 径 设 置 


如 果 按 照 本 书 介绍 的 方法 安装 MAMP， 
文件 就 会 放 在 C:MAMP\binmysql\bin 中 。 


文件 夹 里 。 每 次 都 执行 这 样 的 操作 会 让 人 觉得 
下 面 将 介绍 Creators Update ( 版 本 1703 ) 以 后 的 Windows 10 的 设置 方法 ， 如 明 


























请 通过 控制 面板 打开 “系统 属性 ”对 话 框 。 
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以 后 使 用 的 MySQL 监视 器 (MySQL monitor ) 的 程序 











因此 ， 要 想 启 动 MySQL 监视 带 ， 就 需要 先 移 动 到 这 个 
有 些 麻烦 ， 所 以 我 们 事先 设置 一 下 相关 路 径 。 

















是 其 他 版 本 ， 





J 右键 点 击 “开始 ”一 “系统 ”一 “相关 设置 ”中 的 “系统 信息 ”一 “高 级 系统 设置 "。 


@) 点 击 画面 下 方 的 “环境 变量 ”( 见 图 





2-15 )。 





系统 属性 
计算 机 名 硬件 ”高级 。 系统 保护 远程 


要 进行 大 多 数 更 改 ， 你 必须 作为 管理 员 登录 。 
性 能 
视觉 效果 ， 处 理 器 计划 ， 内 存 使 用 ， 以 及 虚拟 内 存 


用 户 配置 文件 
与 登录 帐户 相关 的 桌面 设置 


启动 和 故障 恢复 
系统 启动 、 系 统 故障 和 调试 信息 


确定 














设置 (5).. 





设置 (E). 


设置 人- 


环境 变量 (N)- 


取消 应 用 (A) 


有 点 











图 2-15 














HH 


由 
Tu 








“系统 属性 ”对 话 框 
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@) 选择 “系统 变量 ”中 的 变量 “Path”， 点 击 “ 编 辑 ”按钮 ( 见 图 2-16 )。 









































环境 变量 x 
brtang 的 用 户 变量 (U) 

变量 值 
Path CA\Users\brtang\AppData\Local\Programs\Python\Python37\Scrip... 
PT7HOME CNProgram Files (x86)\Cisco Packet Tracer 7.0 
TEMP Ci\Users\brtang\AppData\Loca\Temp 
TMP Ci\Users\brtangAppData\LocaN\Temp 

新 建 (N)- 编辑 (6)- 删除 (D) 

系统 变量 (S) 
ED 值 ^ 
ComSpec CA\WINDOWS\system32\cmd.exe 
DEFLOGDIR C\ProgramData\McAfee\Endpoint Security\Logs 
NUMBER OF PROCESSORS 4 
OS Windows NT 、 
a Oo 选择 “Path” 

PATHEXT ‘COM;.EXE;.BAT;.CMD;.VBS;.VBE;.JS;JSE;, WSF;.WSH;.MSC 
PERL5UB 
LPROCFSSOR_ARCHITFCTURF _AMD64 x 

新 建 (W)… 编辑 ()- 删除 (L) 

点 击 这 里 

















图 2-16 ”环境 变量 
@ 点击“ 新建” 按钮, 输入 C:MAMP\bin\mysql\bin ( 见 图 2-17 )。 另外， 根据 Windows 的 版 
本 ， 有 时 会 在 当前 变量 值 的 最 后 加 上 “;”， 然 后 再 追加 C:MAMP\bin\mysql\bin。 





















































| 编辑 环 
| |%SystemRoot%\system32 新 建生 一 点 击 这 里 
| | %SystemRoot% 
| | %SystemRoot%\System32\Wbem 编辑 (E) 
%SYSTEMROOT%\System32\WindowsPowerShell\v1.0\ 
C\MAMP\bin\mysqlN\bin 加 而 
删除 [D) = 
输入 需要 添加 的 环境 变量 
上 移 (U) 
| 下 移 (D) 
| 
| 编辑 文本 由. 














el | 





图 2-17 编辑 环境 变量 
@) 点 击 “确定 "， 关 闭 各 个 对 话 框 。 
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2.4 关于 中 文 的 设置 


某 种 字符 的 集合 称 为 字符 集 (character set )。 例 如 中 文 汉 字 ”和 英文 字母 等 都 是 字符 集 的 一 种 。 

在 计算 机 和 网 络 上 表示 字符 时 ， 需 要 给 各 个 字符 添加 识别 信息 ， 例 如 数字 或 者 符号 等 。 给 各 个 
字符 分 配 的 数字 或 者 符号 称 为 字符 代码 ( character code )。 

给 字符 集 分 配 字符 代码 时 的 规则 称 为 字符 编码 ( character encoding )， 中 文 相关 的 主流 字符 编码 
如 表 2-2 所 示 。 

















表 2-2 中 文 相关 的 主流 字符 编码 


字符 编码 名 说 明 

























































































































































































ISO-8859-1 字母 的 字符 编码 ( 单字 节 编 码 ) 

GB 2312 1981 年 5 月 1 日 发 布 的 简体 中 文 汉 字 编码 国家 标准 

GBK 1995 年 12 月 发 布 的 汉字 编码 国家 标准 ， 是 对 GB 2312 编码 的 扩充 

cp936 微软 Windows 环境 上 在 GB 2312 的 基础 上 扩展 的 字符 编码 ， 通 常 被 视 为 等 同 于 GBK 

EUC-CN Linux/Unix 环境 上 以 GB 2312 为 基础 的 字符 编码 

ED 2000 年 3 月 17 日 发 布 的 汉字 编码 国家 标准 ， 是 对 GBK 编码 的 扩充 ， 兼容 GBK 和 GB 2312 
字符 集 

UTF-8 针对 Unicode 字符 集 的 一 种 可 变 长 度 字符 编码 


























举例 来 说 ,“ 啊 ” 字 在 中 国 国 家 标准 GB 2312、GBK 和 GB 18030 字符 编码 中 的 编码 都 为 
“B0A1”， 而 在 UTF-8 中 则 为 “E5958A”。 从 中 可 以 看 出 ， 即 使 文字 相同 ， 如 果 字 符 编码 不 同 ， 表 
示 的 字符 也 可 能 不 同 ， 由 此 才 出 现 了 乱码 现象 。 

世界 上 有 许多 语言 系统 和 文字 ， 所 以 除了 上 面 介绍 的 内 容 之 外 还 有 无 数 的 字符 集 和 字符 编码 。 
但 这 样 一 来 ， 面 向 全 世界 发 布 的 网 页 等 就 会 变 得 很 难处 理 ， 因 此 人 们 开发 了 可 以 支持 全 世界 所 有 字 
符 的 字符 集 Unicode。Unicode 的 字符 编码 方式 有 好 几 种 ， 现 在 最 为 常用 的 是 在 HTML 5 中 被 默认 
使 用 的 UTF-8。 顺 便 说 一 下 ， 在 UTF-8 的 情况 下 ， 字 符 集 和 字符 编码 都 称 为 UTF-8。 

另外 ， 很 多 时 候 字 符 编码 也 用 “字符 代码 ”这 个 词 来 表示 ， 不 过 本 书 统一 使 用 “字符 编码 ”这 
个 名 称 。 










































































本 书 中 ,存储 在 MySQL 中 的 数据 使 用 了 字符 编码 UTF-8。 这 主要 是 为 了 方便 和 PHP 结合 起 来 











(D 原版 书 主要 对 上 日语 的 相关 内 容 进 行 了 介绍 ， 为 了 方便 国内 读者 阅读 ， 在 此 进行 了 适当 的 调整 ， 增 加 了 中 文 的 相关 


内 容 。 译 者 注 
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开发 Web 应 用 程序 。 通 过 Web 应 用 程序 ， 我 们 可 以 把 输入 的 数据 存储 到 数据 库 ， 再 从 数据 库 读 出 
数据 显示 在 Web 画面 上 。 在 Web 的 世界 中 默认 使 用 的 字符 编码 是 UTF-8， 所 以 如 果 数 据 库 的 字符 
编码 也 是 UTF-8， 就 不 需要 进行 字符 编码 的 转换 了 。 
而 本 书 在 介绍 数据 库 的 基本 操作 时 ,使 用 了 运行 在 命令 行 上 的 一 个 名 为 MySQL 监视 器 (一 3.2 
) 的 客户 端 软件 。 因 为 中 文 Windows 系统 中 命令 行 的 默认 字符 编码 是 GBK， 所 以 MySQL 监视 
器 的 输入 输出 字符 串 的 字符 编码 也 是 GBK。 也 就 是 说 ， 如 图 2-18 所 示 ， 存 储 在 数据 库 中 的 字符 的 
字符 编码 和 客户 端的 输入 输出 中 使 用 的 字符 编码 并 不 相同 。 































































































MySQL 监 视 器 MySQL 





PHP 文 件 
( UTF-8) 








UTF-8 











Apache 





图 2-18 本 书 字符 编码 的 处 理 
在 这 种 情况 下 ， 当 向 数据 库存 储 数据 时 (GBK 一 UTF-8 )， 以 及 从 数据 库 读 取 数 据 显 示 在 客户 
端 上 时 (UTF-8 一 GBK )， 都 需要 进行 字符 编码 的 转换 。 不 过 请 放心 ， 转 换 操作 并 不 麻烦 ， 只 要 在 
MySQL 的 设置 文件 中 设置 一 番 即 可 。 
此 外 ， 关 于 本 书 使 用 的 文本 文件 的 字符 编码 ， 还 有 几 个 需要 注意 的 地 方 ， 在 以 后 的 介绍 中 我 会 
适时 进行 提醒 。 



































2.4.2 ”修改 my.ini 








MySQL 中 字符 编码 的 初始 设置 可 以 通过 文本 文件 my.ini 进行 修改 。 如 果 根 据 本 书 2.2 节 的 介 
绍 安装 了 MAMP，my.ini 就 会 保存 在 下 面 的 文件 夹 中 。 





























C:\MAMP\conf\mysql 
































用 文本 编辑 器 打开 my.ini， 然 后 添加 代码 清单 2-1 中 红字 部 分 的 内 容 。 国 在 第 39 行 左 右 ， 
用 于 设置 MySQL 数据 库 的 字符 编码 。 回 在 第 146 行 左右 ， 用 于 设置 mys ql 命令 (后 述 ) 的 字 
符 编 码 。 
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代码 清单 2-1 修改 my.ini 





[mysqld] 


port = 3306 
socket = mysqgl 


skip-external-locking 
key buffer size = 16M 


myisam sort buffer size = 8M 

basedir = C:/MAMP/bin/mysql/ 

datadir = C:/MAMP/db/mysql/ 

Character-set-server=utf8 


[mysql] 

no-auto-rehash 

# Remove the next comment character if you are not familiar with SQL 
#safe-updates 

default character set=gDk 











修改 完 myini 后 保存 ， 并 在 MAMP 的 画面 上 重新 启动 服务 器 。 
3.3.4 节 将 介绍 如 何 确认 是 否 正 确 地 设置 了 字符 编码 ， 如 果 没 有 正确 地 进行 设置 ， 请 再 重新 确认 
一 下 上 面 的 操作 。 


















































2.5 ”本 书 中 使 用 的 表 


本 书 中 作为 示例 使 用 的 数据 库 ， 其 名 称 全 部 为 dp1。 另 外 ,介绍 MySQL 时 使 用 的 表 ， 主 要 是 
虚拟 公司 “D 公司 ”的 销售 信息 表 也 和 员工 信息 表 tb1。 








人 2.5.1 销售 信息 表 tb ( 按 员工 号 统计 的 月 销售 额 ) 


表 2-3 是 按 D 公司 员工 号 统计 的 每 月 销售 数据 。 
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表 2-3 DD 公司 2018 年 第 2 季度 销售 信息 















































员工 号 销售 额 ( 万 元 ) 月 份 

empid sales month 

A103 101 4 

A102 54 5 

A104 181 4 

A101 184 4 

A103 17 5 

A101 300 5 

A102 205 6 

A104 93 5 

A103 12 6 

A107 87 6 

假设 销售 额 以 万 元 为 单位 。 也 就 是 说 ， 员 工 号 为 A103 的 员工 在 4 月 份 的 销售 额 是 101 万 元 。 

在 对 表 进 行 操作 的 时 候 想 想 身边 的 销售 公司 ， 就 很 容易 理解 了 。 





本 书 中 ， 列 名 (一 1.1.2 节 ) 使 用 了 empid (员工 号 )、sales (销售 额 )、month (月 份 ) 这 样 的 
英文 表示 。 对 于 列 名 、 表 名 和 变量 名 等 的 命名 方法 ， 虽 然 很 多 程序 员 有 自己 惯用 的 规则 ， 但 为 了 更 
容易 理解 ， 本 书 采用 了 简单 的 英文 字母 作为 列 名 。 








人 2.5.2 员工 信息 表 tb1 ( 各 员工 号 代表 的 员工 的 姓名 、 年 龄 ) 


tbl 是 DD 公司 的 员工 信息 表 ( 见 表 2-4 )， 通 过 这 个 表 我 们 可 以 知道 ， 刚 才 介绍 的 表 tb 中 销售 额 
为 101 万 元 、 员 工 号 为 Al103 的 员工 是 中 川 ， 年 龄 是 20 岁 。 





表 2-4 DD 公司 2018 年 的 员工 信息 




















员工 号 姓名 年 龄 
empid name age 
A101 佐 蔷 40 
A102 高 桥 28 
A103 中 川 20 
A104 渡 边 23 
A105 泽 35 












































另外 ,在 后 面 的 解说 中 表 tpl 的 内 容 会 发 生变 更 ， 所 以 为 了 使 没有 从 头 开 始 阅读 本 书 的 读 
者 不 会 在 理解 上 产生 问题 ， 我 会 以 操作 和 表 tbl 内 容 完 全 一 样 的 tbl1A、tb1B、tb1C 等 的 形式 进 
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行 介绍 。 这 样 既 可 以 创建 多 个 内 容 相同 的 表 ， 也 可 以 忽略 中 途 修改 的 内 容 ， 使 用 相同 的 表 tbl 
行 


关于 表 的 复制 请 参考 7.1 节 的 内 容 。 


本 章 介绍 了 以 下 内 容 。 


@ MAMP 的 安装 

@ 到 MySQL 启动 为 止 的 设置 

@ MySQL 中 中 文字 符 编码 的 设置 

@ 书 中 使 用 的 员工 信息 表 tb1 和 销售 信息 表 tb 的 内 容 

















如 果 MySQL 安装 失败 ， 就 很 难 学 习 第 3 章 以 后 的 内 容 了 。 在 安装 失败 的 情况 下 ， 请 参考 附录 2， 
提前 把 问题 解决 掉 。 


> 自我 检查 
下 面 检查 一 下 本 章 学 习 的 内 容 是 否 全 部 理解 并 掌握 了 。 


口 能 够 正常 安装 MAMP 

口 能 够 设置 字符 编码 

口 能 够 设置 MySQL 的 路 径 、 扩 展 名 等 

口 理解 MySQL 里 中 文 的 相关 处 理 

口 理解 员工 信息 表 tb1 和 销售 信息 表 tb 的 内 容 





关于 macOS 版 的 MAMP 


MAMP 原本 就 是 面向 macOS 开发 的 软件 ，Windows 版 是 之 后 才 开 发 的 。 现 在 依然 按照 这 种 开发 顺序 进行 
发 ， 所 以 本 书 使 用 的 Windows 版 的 MAMP 的 版 本 是 3.3.1， 而 macOS 版 本 已 经 是 4.2 了 。( 2017 年 8 月 时 
的 信息 ) 

Windows 版 的 MAMP 和 macOS 版 的 MAMP 有 以 下 几 个 不 同 点 。 需 要 注意 ， 这 些 内 容 是 在 macOS 版 
MAMP 4.2 的 基础 上 总 结 出 来 的 ， 将 来 很 有 可 能 发 生 改 变 。 























































































































四 截至 2018 年 3 月 ，Windows 版 的 MAMP 最 新 版 本 为 MAMP & MAMP PRO 4.0， 而 macOS 版 的 MAMP 最 新 版 本 
为 MAMP & MAMP PRO 4.4.1。 译 者 注 
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@ MAMP 和 MAMP PRO 
macOSs 版 的 MAMP 和 MAMP PRO 会 被 同时 安装 。 当 使 用 标准 版 的 MAMP 时 ， 请 启动 “应 
MAMP” 中 的 MAMP.app。 






































程序 > 




















@@ mysql 命令 的 路 径 
用 于 显示 MySQL 监视 器 的 mysql 命令 在 路 径 “ 应 用 程序 > MAMP > Library > bin” 中 。 可 以 通过 在 终端 
执行 下 面 的 export 命令 来 添加 这 个 命令 的 搜索 路 径 。 
























































export PATH=$PATH:/Applications/MAMP/Library/bin 





@ htdocs 文件 夹 的 路 径 
文件 根 目录 的 htdocs 文件 夹 的 路 径 变 为 “应 用 程序 > MAMP > htdocs”。 




















人 @@ 使 用 的 端口 
macOS 版 的 MAMP 中 包含 的 Apache 和 MySQL 使 用 的 端口 与 Windows 版 的 有 所 不 同 ， 这 是 因为 macOS 
上 默认 安装 了 Apache， 为 避免 发 生 冲突 才 使 用 了 不 同 的 端口 。 


macOS 版 MA/Tato lo 














































































































Apache 的 端 8888 号 80 号 
MySQL 的 端 8889 号 3306 号 
姑 此 ， 当 访问 macOS 版 Apache 管理 的 Web 画面 时 ， 需 要 像 下 面 这 样 在 URL 上 指定 端口 。 


















































© http:/localhost:8888/MAMP 


第 2 部 分 
MySOQL 的 基础 知识 


ED MySQL 监视 器 
EL 创建 数据 库 


ED 数据 类 型 和 数据 输入 


EE 复制 、 册 除 表 和 记录 




















局 | 


我 们 将 从 MySQL 的 基础 内 容 开始 进行 实践 ， 其 中 包括 创建 数据 库 ， 创 建 表 ， 插 入 数据 ， 
确认 数据 ， 以 及 修改 、 复 制 和 删除 表 的 方法 。 首 先是 数据 库 的 入 门 篇 ， 在 这 部 分 内 容 中 ， 我 
们 会 学 习 数 据 库 操作 的 一 系列 流程 。 刚 开始 接触 数据 库 的 读者 不 妨 每 天 做 一 下 附录 3。 

大 家 知道 成 功 的 秘诀 是 什么 吗 ? 那 就 是 在 成 功 之 前 绝 不 放弃 。 让 我 们 一 起 加 油 吧 ， 
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第 3 草 MySQL 上 监视 器 











本 章 我 们 将 从 MySQL 的 基础 内 容 开 始 讲 起 。 

操作 MySQL 的 方法 有 好 几 种 ， 首 先 我 1 
视 器 的 操作 方法 。 学 会 使 
人 


全 















































门 来 学 习 MySQL 监 
用 MySQL 监视 器 ， 就 可 以 执行 各 种 命 
地 对 MySQL 进行 操作 了 。 
























































3.1 创建 数据 库 前 的 准备 事项 


























首先 ， 我 们 要 确认 一 下 MySQL 的 使 用 环境 是 否 都 准备 好 了 。 需 要 提前 完成 的 工作 如 下 所 示 。 
QD 安装 好 MySQL 

@) 设置 好 MySQL 安装 文件 夹 的 路 径 

@@ 完成 MySQL 的 中 文 设置 




















如 果 还 没有 安装 好 MySQL ， 请 参考 第 2 章 的 内 容 进 行 安装 和 设置 。 





3.2 什么 是 MySQL 监视 器 


安装 好 MySQL 之 后 ， 就 可 以 使 用 被 称 为 MySQL 监视 器 的 程序 了 。MySQL 监视 器 这 个 客户 端 
程序 以 用 于 操作 MySQL 的 CUI ( Character User Interface ， 字 符 用 户 界面 ) 为 基础 。MySQL 监视 器 
不 支持 使 用 鼠标 来 进行 点 击 或 拖 忠 ， 用 户 需 要 使 用 键盘 直接 输入 命令 来 操作 数据 库 ， 输 出 结果 也 全 
部 以 文本 的 形式 显示 出 来 。 


初次 使 用 命令 行 操作 数据 库 的 读者 ， 在 习惯 使 

































































| 键盘 操作 数据 库 之 前 可 能 会 很 辛 苗 。 不 过 ， 
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“自己 敲 键盘 发 出 命令 ， 然 后 返回 结果 ”其 实 是 一 件 非 常 令 人 愉快 的 事 ， 我们 要 通过 反复 操作 来 逐 
浙 习 惯 这 种 操作 方式 。 


3.3 启动 MySQL 监视 器 


人 3.3.1 启动 终端 软件 


下 面 我 们 使 用 MySQL 监视 器 操作 MySQL。MySQL 监视 器 需要 使 用 终端 软件 ， 例 如 Windows 
环境 的 “命令 提示 符 ” 或 “Windows PowerShell”、macOS 和 Linux 环境 的 “终端 ”( terminal ) 等 。 
所 以 ， 大 家 要 先 学 会 使 用 作为 命令 操作 平台 的 命令 提示 符 和 终端 。 本 书 将 以 Windows 的 命令 提示 
符 为 例 进 行 介 绍 。 

如 图 3-1 所 示 ，Windows 的 命令 提示 符 需 要 通过 选择 “开始 ”菜单 的 “Windows 系统 ”一 “ 命 
令 提 示 符 ”启动 (在 Windows 7 的 情况 下 ， 选 择 “ 开 始 ” 一 “所 有 程序 ”一 “附件 ”一 “命令 提示 
符 ”)， 或 者 同时 按 下 Win 键 和 R 键 ， 显示 出 “运行 ”对 话 框 后 ,输入 “cmd”， 点 击 “ 确 定 ”"， 这 样 
也 可 以 启动 命令 提示 符 。 
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Create 


Windows Defender 安全 中 心 


贺 | windows Media player 图 
加 图 


| powershal 
Calendar 


(me) | 


我 的 Office OneNote 


| Windows 附件 


中 9S 


Skype 


winscp 
x 


两 :te 
踢 DO D 
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这 里 输入 














图 3-2 命令 提示 符 


本 书 虽 然 以 Windows 的 命令 提示 符 为 例 进行 说 明 ， 但 是 在 macOS 或 Linux 的 终端 进行 操作 时 ， 


步骤 也 基本 相同 。 

画面 的 左 侧 会 显示 出 “>” 图 标 ， 这 个 图 标 叫 作 提示 符 ( prompt )， 用 于 表示 正在 等 待 用 户 输入 
命令 的 状态 。C:NUsersmisi 是 用 户 当前 所 在 的 文件 夹 位 置 ( 当前 路 径 )。 这 个 当前 路 径 会 根据 具体 使 
用 的 环境 而 发 生 改变 。 





















































在 命令 提示 符 里 输入 mysql 命令 ， 就 可 以 启动 MySQL 监视 器 了 。 具 体格 式 如 下 。 
启动 MySQL 监视 器 




















用 户 名 ”和 “-p 密码 ” 称 为 选项 ( option )。 在 上 面 的 例子 中 ， 执 行 mysql 命令 的 同时 也 
肯定 了 用 户 名 和 密码 。 如 果 没 有 设置 密码 ， 则 不 需要 输入 “-p 密码 ”选项 。 

“mysql” 和 “-u” 之 闻 ， 以 及 “用 户 名 ”和 “-p” 之 间 ， 都 需要 输入 半角 空格 。 但 是 “-p” 
和 “密码 ”之 间 不 能 有 空格 。 注 意 ,“-p” 和 “密码 ”之 间 有 空格 会 发 生 错误 。 

在 通过 MAMP 安装 的 MySQL 中 ， 初 始 设置 的 用 户 名 和 密码 如 下 。 







































































用 户 名 一 ”root 


密码 一 root 
































一 般 来 说 ， 这 种 简单 的 密码 会 在 安全 方面 存在 很 大 的 问题 ， 所 以 通常 需要 修改 密码 或 者 在 缩小 
权限 后 创建 一 个 新 用 户 。 但 为 了 尽早 开始 MySQL 的 学 习 ， 本 书 省 去 了 这 部 分 操作 。 
下 面试 着 启动 一 下 MySQL 监视 器 。 在 命令 提示 符 中 输入 下 面 的 命令 。 


mysql -u root -proot 
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3.3 启动 MySQL 监视 器 




















启动 MySQL 监视 右 后 ， 如 果 显 示 出 Welcome to the MySQL monitor. Commands end 
with ; or \g. 之 类 的 消息 ,就 意味 着 成 功 连接 了 MySQL ( 见 图 3-3 )。 























启动 MySQL 
监视 器 











the _ current input s 





显示 出 提示 符 
mysq 

















3-3 ”启动 MySQL 监视 器 





在 显示 的 信息 中 ，commands end with ; or \g. 表示 请 在 接 下 来 输入 的 命令 的 最 后 加 上 
“;” 或 “\g”。 在 本 书 中 ，MySQL 监视 器 的 命令 的 最 后 会 统一 加 上 “;”。 
命令 提示 符 看 起 来 和 最 初 的 状态 没有 什么 两 样 ， 但 最 下 方 显示 出 了 MySQL 监视 器 的 提示 符 


mysql， 后 面 的 光标 也 变 得 一 闪 一 闪 的 。 我 们 要 在 这 里 输入 SQL 语句 的 命令 。 











命令 提示 符 的 UTF-8 设置 
本 书 以 “命令 提示 符 端 使 用 字符 编码 GBK 进行 数据 的 输入 和 输出 ， 而 数据 库 则 使 用 UTF-8 存储 数据 ”为 前 


























































































































其 实 ， 我 们 也 可 以 把 命令 提示 符 端的 字符 编码 设置 成 UTF-8。 为 了 达到 这 个 目的 ， 我 们 需要 按照 下 面 的 格式 


~y 


E 命 令 提示 符 中 执行 chcpo 

















任 





O 


ep 是 6S500 下 
































JE 人 小 


chcp 命令 修改 命令 提示 符 的 字符 编码 ，65001 代表 UTF-8。 如 果 这 样 设置 ， 就 不 需要 修改 my.ini 中 加 
的 办 3 222 
但 是 ， 如 果 采 用 了 这 种 方法 ， 就 需 遍 一 遍地 执行 这 个 命令 ， 在 某 些 环境 中 还 可 能 会 发 生男 


情况 ， 所 以 本 书 没有 采 这 种 方法 。 








显示 毅 溃 的 
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@ 3.3.3 当 MySQL 监视 器 无 法 启动 时 

如 果 没 有 显示 “Welcome...” 消 息 ， 发生 错 误 ， 我 们 可 以 从 以 下 几 个 方面 来 考虑 。 
PP 密码 错误 

输入 正确 的 密码 ， 重 新 尝试 连接 。 


P 没有 设置 密码 ， 但 指定 了 -P 选项 
省 略 -p 的 指定 ， 直 接 执行 mysql -u root。 





P 路 径 设置 不 正确 
根据 2.3 节 介 绍 的 内 容 正 确 设置 MySQL 路 径 。 另 外 ， 即 使 路 径 设 置 错误 ， 在 命令 提示 符 中 输 
入 cd C:\MAMP\bin\mysql\bin, 移动 所 在 文件 夹 ， 也 能 使 用 相同 的 命令 启动 MySQL 监视 器。 











> “-p” 和 “密码 ” 之 间 有 空格 
二 者 中 间 不 要 输入 空格 。 


区 各 单词 之 间 的 空格 是 全 角 空 格 
各 单词 之 间 的 空格 必须 是 半角 空格 。 





PP -p、root、-u 变 成 了 大 写字 母 
再 次 确认 是 否 所 有 的 字母 都 是 小 写字 母 。 





全 角 空 格 和 半角 空格 
对 于 在 命令 提示 符 和 MySQL 监视 器 中 输入 的 SQL 语句 以 及 PHP 脚本 (一 15.7.3 节 )， 其 中 的 空格 必须 是 
角 空 格 。 在 入 门 阶段 ， 许 多 人 会 不 小 心 输入 全 角 空 格 ， 以 致 错误 发 生 。 






























































S 3.3.4 确认 MySQL 中 字符 编码 的 设置 情况 

下 面 我 们 试 着 在 MySQL 监视 器 启动 的 状态 下 确认 MySQL 中 字符 编码 的 设置 情况 。 在 MySQL 
监视 器 中 执行 status 命令 ， 字 符 编码 设置 等 信息 就 会 显示 出 来 。 

启动 MySQL 监视 器 ,输入 status， 然 后 按 Enter 键 ， 就 能 在 本 书 使 用 的 环境 下 看 到 以 下 执 
行 结果 。 
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mysql Ver 14.14 Distrib 5.6.34, for Win32 (AMD64) 


Connection id: 4 
Current database: 


Current USBer': root@localhost 

SSL: Not in use 

Using delimiter: ; 

Server version: 5.6.34-1og MySQL Community Server (GPL) 
Protocol version: 10 

Connection: localhost via TCP/IP 


Server characterset: utf8 
Db characterset: EtS 人 
Client characterset: gbk 
Conn. characterset: gbk 一 一 一 一 一 一 一 咎 
TCR Port 3306 


Uptime: 8 min 37 sec 





Threads: 1 Questions: 15 Slow queries: 0 Opens: 70 Flush tables: 1 Open 
tables: 63 Queries per second avg: 0.029 

















国 表 示 服 务 器 端的 字符 编码 设置 ， 回 表示 客户 端的 字符 编码 设置 。 请 确认 显示 的 内 容 是 否 和 示 
例 一 致 。 

另外 ,字符 编 码 的 设置 情况 还 可 以 通过 在 MySQL 监视 器 中 输入 SHOW VARIABLES LIKE 
'char%'; 来 确认 。 
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人 3.4. 退出 MySQL 监视 器 


接 下 来 介绍 一 下 退出 MySQL 监视 器 的 方法 。 
在 提示 符 状态 下 ， 我 们 可 以 通过 输入 exit 或 者 quit， 然后 按 下 Enter 键 来 退出 MySQL 监 
视 器 。 
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退出 MySQL 监视 器 


退出 MySQL 监视 器 后 会 返回 到 命令 提示 符 “>”。 退 出 命令 提示 符 或 者 终端 时 也 要 执行 exit。 
不 过 这 里 先 不 要 退出 命令 提示 符 。 


退出 命令 提示 符 ( 终端 ) 


; exit ( 或 者 直接 关闭 窗口 ) 

















ae 使 用 历史 命令 


试 着 再 次 启动 MySQL 监视 器 。 我 们 可 以 像 3.3.2 节 介 绍 的 那样 通过 输入 mysql -u root 
-proot 来 启动 MySQL 监视 器 ， 也 可 以 使 用 下 te 

在 没有 退出 命令 提示 符 或 者 终端 的 状态 下 ， 试 着 按键 盘 上 的 1 ， 看 看 会 显示 出 什么 内 容 ( 见 
图 3-4 )。 





















































使 用 1 显示 输入 历史 

















图 3-4 使 用 1 的 输入 历史 功能 








在 命令 提示 符 或 者 终端 上 输入 的 命令 ,通常 会 作为 输入 历史 残留 下 来 。 所 以 ,我 们 可 以 通过 1 
键 和 | 键 把 过 去 的 输入 历史 按 顺序 显示 出 来 。 另 外 ， 按 下 F7 键 后 ， 历 史 命令 会 全 部 显示 出 来 ,我 
们 可 以 从 中 选择 想 要 执行 的 命令 。 












































如 上 所 述 ， 使 用 1 键 将 之 前 输入 的 mysql -u root -proot 显示 出 来 ， 然 后 按 Enter 键 ， 就 
可 以 立即 启动 MySQL 监视 器 了 ， 但 是 这 种 操作 方式 也 存在 安全 隐患 。 在 没有 退出 终端 的 情况 下 ， 
如 果 其 他 人 使 用 了 这 台 计 算 机 ，-p 中 设置 的 密码 就 会 被 别人 看 见 。 

所 以 这 一 次 我 们 试 着 用 防止 非法 侵入 的 方法 启动 MySQL 监视 器 。 试 着 只 输入 mysql -u 
root -p 命令， 你 就 会 看 到 以 下 要 求 输入 密码 的 提示 。 
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3.5 设置 MySQL 管理 员 root 的 密码 





>mysql -u root -p 


Enter password: 

















在 这 里 输入 “root” 之 类 的 密码 ， 就 可 以 像 前 面 一 样 启动 MySQL 监视 器 了 。 该 方法 可 以 避免 
通过 命令 历史 盗 取 密 码 ， 因 此 ， 在 一 般 情 况 下 它 不 失 为 一 种 好 的 选择 。 

另外 ,不 仅 是 命令 提示 符 的 命令 ，MysQL 监视 器 中 的 命令 也 可 以 通过 人 键 和 | 键 来 显示 输入 
历史 。 不 管 怎 样 ， 大 家 都 要 记 住 输入 的 命令 会 被 保留 下 来 这 一 点 。 


mysql 命令 的 选项 
在 启动 MySQL 监视 器 的 时 候 ， 我 们 输入 了 mysql -u root -proot, 这 个 命令 其 实 就 是 在 mysql 命令 
的 基础 上 添加 了 -u 和 -p 两 个 选项 。 像 这 样 通过 给 命令 添加 选项 ， 就 可 以 指定 命令 完成 设置 的 各 种 处 理 了 。 

给 mysql 命令 设置 选项 的 方法 主要 有 以 下 两 种 。 



























































db 给“-” 指 定 选项 名 ， 然 后 加 上 设置 的 值 
例 : -u root ( 在 这 种 情况 下 ， 选 项 名 只 占 1 个 字符 ) 


























@ 在 “-- 选项 名 =” 后 加 上 设置 的 值 


例 : mysgqln User=root® password=root 








例如 ， 在 指定 字符 编码 启动 MySQL 监视 器 的 情况 下 ， 可 以 写成 mysql -u root -proot --default- 


character-set=gbko 





3.5 设置 MySQL 管理 员 root 的 密码 


在 MAMP 的 初始 设置 中 ， 管 理 员 root 的 密码 为 “root"。 为 了 便于 介绍 ， 本 书 默 认 使 用 了 这 个 
用 户 名 和 密码 。 下 面 就 来 介绍 一 下 修改 密码 的 方法 。 这 些 操作 了 解 即 可 ， 不 需要 实际 执行 。 























仿 3.5.1 修改 root 用 户 的 密码 

我 们 来 通过 一 个 示例 看 一 下 修改 密码 的 方法 。 假 设 把 密码 修改 为 “1234"。 虽 然 这 个 密码 也 存 
在 安全 问题 ， 但 这 里 仅 作为 修改 密码 的 示例 使 用 ， 还 请 大 家 见谅 。 

启动 MySQL 监视 器 ， 把 下 面 的 命令 输入 到 “>”( 提示 符 ) 的 后 面 ， 然 后 按 Enter 键 。 
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SET PASSWORD FOR root@localhost=PRASSWORD ('!12345) ; 


mysql> SET PASSWORD FOR root@localhost=PASSWORD('1234'); 
Query OK, 0 rows affected (0.00 sec) 














这 样 ，root 的 密码 就 改 成 了 “1234”， 这 时 就 可 以 退出 MySQL 监视 器 了 。 


352 修改 MAMP 的 证 


在 修改 了 root 用 户 密码 的 情况 下 ， 如 果 不 修改 MAMP 的 设置 文件 内 容 ， 就 无 法 退出 和 启动 
MySQL 服务 髓 。 

用 文本 编辑 器 打开 CNMAMP 文件 夹 中 的 MAMP.exe.config 文件 ， 将 代码 清单 3-1 中 的 红字 内 
容 (第 9 行 附近 )， 由 root 改 为 1234。 


代码 清单 3-1 MAMP.exe.config 












































<add key="StopMysqlCommand" value="--user=root --password=1234 --port={0} 
shutdown" /> 




















保存 文件 后 ， 在 MAMP 画面 上 点 击 “Stop Servers”"， 然 后 确认 Apache 和 MySQL 是否 已 经 正 
常 停止 。 再 次 点 击 “Start Servers”"，Apache 和 MySQL 就 会 启动 起 来 。 























B35.3 修改 phpMyAdmin 的 设置 


修改 了 MySQL 管理 员 root 的 密码 后 ， 如 果 要 使 用 phpMyAdmin (一 附录 1 )， 就 需要 修改 
phpMyAdmin 的 设置 文件 。 
用 文本 编辑 器 打开 C:MAMP\binphpMyAdmin 文件 夹 中 的 config.inc.php 文件 ， 对 代码 清单 3-2 
的 红字 内 容 (第 61 行 附近 ) 进行 修改 。 


代码 清单 3-2 ”config.inc.php 











scfg['Servers'] [$i] ['password'] = '1234'; // MySQL password 











本 章 介绍 了 以 下 内 容 。 

@ MySQL 监视 器 的 使 用 方法 

@ 连 接 和 退出 MySQL 的 方法 

@ 历 史 命 令 的 相关 内 容 

@ 修 改 密码 的 方法 

自我 检查 

下 面 检查 一 下 本 章 学 习 的 内 容 是 否 全 部 理解 并 掌握 了 。 
口 能 够 启动 和 退出 MySQL 监视 器 


口 能 够 在 不 泄露 密码 的 情况 下 启动 MySQL 监视 器 
口 能 够 利用 选项 启动 MySQL 监视 器 








口 能 够 使 用 历史 命令 
i 练习 题 
问题 1 








































































































下 面 介绍 了 如 何在 MySQL 监视 器 上 创建 新 用 户 以 及 赋予 用 户 全 部 的 访问 权限 。 请 使 用 该 方法 创建 
有 数据 库 db1 所 有 的 访问 权限 、 用 户 名 为 “test"、 密 码 为 “1234” 的 用 户 信息 。 


































































































“用 户 名 ”需要 按照 “用 户 名 @ 主机 名 ”的 方式 书写 。 本 书 的 所 有 操作 都 在 localhost 中 进行 。 
“赋予 的 权限 ”如 果 是 所 有 权限 ， 就 设置 为 “ALL”; 如 果 仅 允许 SELECT 和 UPDATE， 就 设置 为 
“SELECT, UPDATE”; 如 果 是 所 有 数据 库 的 所 有 表 ， 就 设置 为 “*.*”。 





































































































户 名 ; ”来 删除 用 户 。 











| 这 是 正文 中 未 涉及 的 新 用 户 的 设置 方法 。 另 外 ， 可 以 使 用 root 的 权限 执行 “DROP USER 
【2 
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局 参考 答案 














用 root 启动 MySQL 监视 器 ， 执 行 下 面 的 命令 。 


CREATE USER test@localhost IDENTIFIED BY 


GRANT ALL ON dbl.* TO test@localhost; 


Dg 











mysql> CREATE USER testQ@localhost IDENTIFIED BY '1234'; 
Query OK, 0 rows affected (0.05 sec) 


mysql> GRANT ALL ON dbl.* TO test@localhost; 
Query OK, 0 rows affected (0.02 sec) 








够 正常 启动 。 





mysql -u test -pl234 





注意 本 书 的 有 些 内 容 需 要 有 root 权限 才能 执行 ， 所 以 在 学 习 过 程 中 ， 请 使 月 





PowerShell 和 命令 提示 符 














本 书 以 使 用 命令 提示 符 操作 MySQL 监视 器 
































为 例 进 行 了 介绍 是 命令 


Windows PowerShell。 





提示 符 的 下 一 代 版 本 是 
在 Windows 8.1 或 旧版 




















Windows 10 中 ， 




















右键 点 击 开始 按钮 就 可 以 选 





择 “ 命 令 提 示 符 "， 但 在 最 新 版 的 Windows 10 中 ， 








“命令 提示 符 ” 换 成 了 “Windows PowerShell”。 

















PowerShell 向 前 兼容 命令 提示 符 ， 所 以 我 们 也 
可 以 在 PowerShell 上 启动 MySQL 监视 器 ， 操 作 数 
据 库 。 新 出 现 的 PowerShell 拥有 很 多 高 级 功能 ， 但 
它 和 命令 提示 符 之 间 存 在 很 多 细微 的 差异 ， 我 们 在 


















































使 用 的 过 程 中 要 多 加 注意 。 


Windows P. 

















看 的 内 容 ， 确 认 MySQL 监视 器 是 否 外 


CC 





日 root 进行 操作 。 
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本 章 将 试 着 创建 数据 库 。 我 们 将 学 习 创 建 数据 库 一 创建 表 一 
插入 数据 一 确认 数据 这 一 系列 创建 表 的 流程 。 















































4.1 创建 数据 库 


首先 ， 试 着 创建 一 个 名 为 dbl 的 数据 库 来 存储 本 书 使 用 的 所 有 表 。 

















4.1.1 创建 数据 库 | 


我 们 可 以 使 用 以 下 命令 创建 数据 库 。 
创建 数据 库 











创建 数据 库 的 方法 有 好 几 种 ， 这 里 先 介 绍 最 基础 的 方法 一 一 使 用 CREATE DATABASE 命令 。 

SQL 语句 的 命令 需要 使 用 半角 字符 输入 ， 输 入 全 角 字 符 会 发 生 错误 。 另 外 ， 因 为 MySQL 不 区 
分 大 小 写 ， 所 以 输入 哪 种 都 可 以 。 也 就 是 说 ,CREATE DATABASE、Create Database 和 
create database 代表 相同 的 命令 。 本 书 SQL 语句 的 命令 统一 使 用 大 写字 母 来 记述 。 
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利 
它 


操作 方法 





输入 下 面 的 命令 ,创建 数 据 库 。 














mysql> CREATE DATABASE dbl; 
Query OK, 1 row affected (0.01 sec) 











创建 第 一 个 数据 库 是 一 项 很 重要 的 工作 ， 不 过 MySQL 监视 絮 只 会 给 一 个 简短 的 回复 。 如 果 顺 
地 创建 了 数据 库 ， 执 行 结 果 会 显示 Query OK，1 row affected (0.01 sec) 这 样 的 信息 。 
表示 “查询 成 功 ， 更 改 了 1 行 ,花费 0.01 秒 ”。 显 示 Query OK 就 代表 命令 执行 成 功 。 在 执行 









































SQL 语句 时 ， 一 定 要 确认 结果 中 是 否 显示 了 这 个 Query OK 的 信息 。 


相 











顺便 说 一 下 ， 数 据 库 名 和 表 名 (一 1.1.2 节 ) 在 Windows、macOS 和 Linux 上 的 处 理 方 法 并 不 
同 。 在 Windows 和 macOSs 的 环境 中 不 区 分 字符 的 大 小 写 ， 但 是 在 Linux 环境 中 却 区 分 大 小 写 。 











也 就 是 说 ， 表 tbl 和 表 TB1 在 Windows 和 macOS 环境 中 会 作为 相同 的 表 名 处 理 ， 但 是 在 Linux 环 
境 中 会 作为 不 同 的 表 名 处 理 。 为 了 便于 和 命令 进行 区 分 ， 本 书 基本 上 使 用 了 小 写字 母 来 表示 数据 库 


名 、 














表 名 和 列 名 。 
此 外 ， 虽 然 我 们 也 可 以 使 用 中 文 来 命名 数据 库 名 、 表 名 和 列 名 ， 但 是 这 样 会 增 大 问题 发 生 的 概 




















率 ， 所 以 最 好 不 要 使 用 中 文 。 
> 发 生 错 误 的 情况 


本 








如 果 发 生 错误 ,不妨 先 确认 一 下 命令 的 最 后 是 否 忘 记 输入 “;”( 分 隔 符 ) 在 MySQL 监视 带 
命令 的 最 后 需要 输入 “;”。 如 果 在 没有 输入 “;” 的 情况 下 直接 按 Enter 键 ， 就 会 显示 “- >” 





标记 。 它 表示 命令 还 没有 结束 ， 此 时 只 要 输入 “;” 并 按 下 Enter 键 , 命令 就 会 执行 。 在 熟悉 这 项 操 


作 


问 





前 会 容易 忘记 敲 “;”， 不 过 没有 关系 ， 显 示 “- >” 标 记 后 再 输入 “; ”也 来 得 及 。 
另外 ， 如 果 显 示 了 You have an error in your SQL syntax (你 的 SQL 语句 的 语法 有 
题 ) 之 类 的 信息 ， 请 确认 一 下 命令 的 拼写 是 否 有 错误 。 如 果 显 示 了 Can't create database 
































'dbl'; database exists (dbl 已 经 存在 ， 无 法 再 次 创建 )， 就 表示 数据 库 dbl 已 经 被 创建 过 


了 





。MySQL 中 无 法 创建 多 个 同名 的 数据 库 。 


在 租赁 服务 器 上 使 用 MySQL 
在 使 用 租赁 服务 器 时 ， 数 据 库 名 通常 是 事先 决定 好 的 ， 所 以 并 不 是 在 所 有 的 情况 下 都 需要 创建 数据 库 或 者 可 
以 给 数据 库 随 意 取 名 字 ， 这 一 点 需要 注意 。 
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4.2.1 确认 数据 库 


4.2 ”确认 创建 的 数据 库 | 43 


@ 一 一 一 一 
SHOW 命令 用 于 确认 数据 库 的 信息 。 大 部 分 信息 能 通过 SHOW 命令 显示 出 来 。 我 们 可 以 使 用 下 


























面 的 命令 显示 现在 已 经 存在 的 数据 库 。 
显示 数据 库 一 览 









































mysql> drop database a; 
Query OK, 0 rows affected (0.00 sec) 


mysql> SHOW DATABASES; 


information schema 
db1 

mysql 

performance schema 
test 











5 rows in set (0.00 sec) 








创建 好 的 数据 库 会 显示 出 来 。 可 以 看 到 ， 以 文本 形式 表示 的 方 格 框 架 的 下 方 显示 了 一 些 内 容 ， 

















其 中 包括 刚刚 创建 的 db1。 另 外 ， 最 下 面 的 一 行 会 显示 x x rows 





in set 这 种 输出 行 数 的 信息 。 








当 输 出 内 容 无 法 容纳 在 MySQL 监视 带 的 一 个 屏幕 中 时 ， 我 们 只 要 确 








认 这 个 行 数 信息 就 可 以 了 。 
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@422 test 和 mysql 数 据 库 


我 们 已 经 知道 数据 库 dbl 创建 成 功 了 。 但 与 此 同时 ，test 和 mysql 等 数据 库 也 被 默认 创建 了 出 
来 。 这 些 不 明 数 据 库 到 底 是 什么 呢 ? 

实际 上 ， 在 安装 MySQL 的 时 候 会 自动 创建 一 个 名 为 test 的 数据 库 。 这 个 数据 库 本 身 是 空 的 ， 
里 面 没 有 任何 表 。 也 就 是 说 ， 即 使 用 户 没有 执行 CREATE DATABASE x x 命令 ， 也 可 以 立刻 进行 
数据 库 测试 。 

而 mysql 是 负责 存储 MySQL 各 种 信息 的 数据 库 ， 它 保存 了 管理 用 户 信息 的 表 user 等 (一 4.10 
节 )。 除 此 之 外 ， 它 还 保存 了 存储 MySQL 信息 的 数据 库 及 示例 数据 库 。 
































4.3 ”指定 使 用 的 数据 库 


成 功 创建 数据 库 后 ， 我 们 试 着 在 数据 库 中 创建 表 。 在 MySQL 中 ， 并 不 是 启动 后 就 能 立即 使 用 
数据 库 ， 还 需要 显 式 地 声明 使 用 什么 数据 库 。 不 过 ， 即 使 不 指定 使 用 的 数据 库 ， 也 可 以 启动 MySQL 
监视 上 需 。 

一 些 RDBMS 要 求 在 启动 MySQL 监视 器 的 同时 指定 数据 库 ， 但 MySQL 并 没有 对 此 提出 要 求 。 























®, 4.3.1 指定 数据 库 ER 
我 们 可 以 使 用 use 命令 指定 数据 库 。 


指定 使 用 的 数据 库 












































mysql> use dbl 











Database changed 
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执行 use 命令 后 ， 结 果 中 会 显示 Database changed (数据 库 改变 了 ) 的 信息 。 明 明 没 有 选 
择 任何 数据 库 却 显示 了 “changed”， 这 一 点 有 些 奇怪 ， 不 过 我 们 并 不 需要 在 意 。 


读者 是 否 注 意 到 命令 的 最 后 没有 输入 “;” 呢 ? 在 SQL 语句 中 ， 命 令 的 最 后 通常 需要 输入 
“;”， 但 是 use 不 是 SQL 语句 ， 所 以 不 输入 “; ”也 没有 问题 。 另 外 ， 我 们 可 以 用 \u 来 代 蔡 use， 
使 用 \u abl 命令 指定 数据 库 。 像 这 种 不 需要 输入 “; ”的 命令 ， 本 书 统一 用 小 写字 母 表 示 。 

另外 ， 因 为 MySQL 可 以 在 不 选择 数据 库 的 状态 下 启动 MySQL 监视 器 ， 所 以 我 们 很 容易 忘记 
当前 使 用 的 是 哪个 数据 库 。 这 一 点 需要 我 们 多 加 注意 。 




















基 显示 当前 使 用 的 数据 库 
我 们 可 以 使 用 下 面 的 方法 查看 当前 使 用 的 数据 库 。 
显示 当前 使 用 的 数据 库 




















可 
| DATABASE() | 
PF + 
| abl | 
兴 入 SS + 





1 row in set (0.00 sec) 








选择 数据 库 启动 MySQL 监视 器 
我 们 可 以 直接 指定 数据 库 启动 MySQL 监视 器 。 


在 这 种 情况 下 ， 可 以 像 mysql dbl -u root -proot 这 样 , 通过 有 
mysql 命令 。 



































记 
四 
任 
时 
少 
注 
引 | 
全 








示 符 中 指定 数据 库 名 来 执行 





4.4 创建 表 tb1 

















现在 数据 库 dbl 还 是 空 的 状态 。 下 面 创 建 员工 信息 表 tb1 并 搬入 各 条 数据 。 即 使 操作 失误 导致 
创建 出 来 的 表 乱 七 八 糟 也 没有 关系 ， 因 为 这 些 表 都 可 以 轻松 删除 (一 7.7 节 )。 
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B44:1 列 和 字段 

在 创建 表 时 ， 需 要 事先 指定 输入 数据 的 类 型 是 什么 、 列 名 是 什么 等 。 

数据 库 中 也 有 分 开 使 用 术语 的 情况 ， 比 如 构成 表 的 项 目 称 为 字段 ( field )， 构 成 记录 的 各 项 目 
的 数据 称 为 列 。 本 书 统一 使 用 “ 列 ”这 一 术语 。 











4.4.2 数据 类 型 
列 中 保存 的 数据 的 种 类 称 为 数据 类 型 。INT 表示 能 够 存储 1、2、3 这 样 的 整数 数据 。VARCHAR 
表示 能 够 存储 字符 数据 。 男 外 ，VARCHAR 的 后 面 会 紧 跟 “(10)” 这 样 的 内 容 ， 它 表示 允许 最 多 输 
入 10 个 字符 。 
这 里 先 记 住 INT 和 VARCHAR 这 两 种 数据 类 型 。 关 于 数据 类 型 的 详细 内 容 ， 我 们 会 在 第 5 章 进 
行 说 明 。 























> 创建 的 表 的 结构 
下 面 创建 包含 以 下 3 个 列 的 表 。 如 表 4-1 所 示 ， 各 列 对 数据 类 型 也 有 一 定 要 求 。 























表 4-1 表 tb1 
列 名 empid name age 
数据 类 型 VARCHAR (10) VARCHAR (10) INT 























创建 表 tb1， 其 中 将 列 age 定义 为 整数 ， 将 列 empid 和 列 name 定义 为 允许 最 多 输入 10 个 字符 。 

这 次 设置 的 列 名 分 别 为 empid、name 和 age。 虽 然 表 名 或 列 名 可 以 使 用 中 文 ， 但 是 使 用 中 文 各 
字 会 发 生 各 种 各 样 的 问题 。 因 此 ， 数 据 库 名 、 表 名 或 列 名 等 最 好 不 要 使 用 中 文 。 本 书 的 表 名 、 列 名 
全 部 使 用 了 半角 英文 字母 和 数字 。 














4.4.3 创建 表 


我 们 可 以 使 用 CREATE TABLE 命令 创建 表 。 在 () 内 使 用 空格 分 开 列 名 和 数据 类 型 ， 各 个 列 
之 间 使 用 “, ”分 隔 。 


创建 表 











下 面 我 们 试 着 创建 D 公司 的 员工 信息 表 tb1。 
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> 表 的 结构 


表 tb1 的 列 结构 























empid name age 
VARCHAR (10) VARCHAR (10) INT 
操作 方法 























在 MySQL 监视 器 中 使 用 db1， 在 此 状态 下 输入 下 面 的 命令 。 
































mysql> CREATE TABLE tbl (empid VARCHAR(10) ,name VARCHAR(10) ,age INT) ， 
Query OK, 0 rows affected (0.25 sec) 




















显示 Query OK 就 表示 成 功 。 不 在 命令 的 最 后 输入 “;” 就 无 法 执行 命令 (一 4.1.1  )。 如 果 
发 生 错 误 ， 请 检查 是 否 出 现 了 拼写 错误 ,或 者 是 否 输入 了 全 角 空 格 ， 检 查 完 之 后 再 尝试 执行 。 
另外 ， 当 存在 同名 的 表 时 会 发 生 错误 ， 不 会 蔡 换 之 前 已 经 存在 的 表 。 








使 用 ”把 数据 库 名 括 起 来 


数据 库 名 、 表 名 和 列 名 可 以 用 `` ( 反 引 号 ) 括 起 来 使 用 ， 本 书 就 不 对 这 种 方法 进行 介绍 了 。 另 外 ， 输 入 到 
列 中 的 字符 串 的 值 需 要 用 ' ' ( 单 引号 ) 或 者 "" ( 双 引 号 ) 括 起 来 。 










































































4.5 ”显示 所 有 的 表 


CS 4.5.1 显示 所 有 的 表 


4.2 节 介 绍 过 SHOW 命令 可 以 显示 MySQL 中 各 种 各 样 的 信息 。 而 当 显示 数据 库 中 所 有 的 表 时 ， 
需要 使 用 SHOW TABLES 命令 。 
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显示 所 有 的 表 


操作 方法 


输入 下 面 的 命令 ， 确 认 执行 结果 














是 
| Tables in dbl | 
+--------------- 十 
| tb1 | 
+--------------- 十 





1 row in set (0.00 sec) 








”指定 字符 编码 创建 表 





在 MySQL 中 输入 字符 到 表 中 时 ， 会 因为 各 种 原因 出 现 字符 乱码 的 情况 。 这 时 有 一 个 方法 可 以 
解决 这 个 问题 ， 那 就 是 指定 字符 编码 创建 表 。 


例如 在 指定 UTF-8 创建 表 时 ， 在 “CREATE TABLE ...” 的 命令 中 加 上 cHARSET=utf8 选项 。 
如 果 无 论 如 何 都 解决 不 了 字符 乱码 的 问题 ， 不 妨 尝试 一 下 下 面 这 种 方法 。 

















CRERATE TABLE tb1l (empid VRARRCHRAR (10) ,mame VRARCHRAR (10) ,age INT) CHRARSET=uUtf8， 





注意 ， 本 书 使 用 的 MySQL 的 默认 字符 编码 是 UTF-8， 所 以 不 需要 按照 上 面 的 设置 来 创 妇 








访问 其 他 数据 库 





在 使 用 use 选择 数据 库 的 状态 下 也 能 够 操作 其 他 数据 库 9 
"数据库 名 ”和 “ 表 名 ”用 “.” 
可 以 使 用 下 面 的 命令 。 









































Pb 的 表 。 这 时 可 以 像 “数据 库 名 . 表 名 ”这 样 把 
连接 起 来 。 例 如 ， 当 从 其 他 数据 库 访问 数据 库 db2 中 的 表 table 的 所 有 记录 时 ， 

























































































SELECT * FROM db2.table; 


这 个 命令 在 没有 使 用 use 选择 数据 库 的 状态 下 也 可 以 执行 。 





























4.6 ”确认 表 的 列 结构 


下 面 来 确认 列 的 数据 类 型 等 结构 。 
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用 于 显示 表 的 列 结构 的 命令 是 DESC 或 者 DESCRIBE。 像 这 种 同一 











况 在 SQL 语句 中 很 常见 。 本 书 尽量 采用 了 简单 的 表示 方法 。 








显示 表 的 列 结构 


个 命令 有 多 种 表示 方法 的 情 








下 面试 着 显示 表 tbl 的 列 结构 。 





> 表 的 结构 


> 显示 出 来 的 表 tb1 的 列 结构 























empid name age 
VARCHAR (10) VARCHAR (10) INT 
操作 方法 








输入 下 面 的 命令 ， 显 示 表 tb1 的 列 结构 。 




















DESC tbl; 
执行 结果 
mysql> DESC tbl; 
+------- +------------- +------ +----- 
| Field | Type | Null | Key 
+------- +------------- +------ +----- 
| empiqd | varchar(10) | YES | 
| name | varchar(10) | YES | 
| age | int (11) | YES | 
+------- +------------- +------ +----- 
3 rows in set (0.05 sec) 

















+--------- +------- + 
| Default | Extra | 
+--------- +------- + 
| NUL | | 
| nr | 
| nr | 
+--------- +------- + 
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Null 表示 “人 允许 不 输入 任何 值 "，Default 表示 “如 果 什 么 值 都 不 输入 就 用 这 个 值 ”( 一 6.11 节 )。 
Field 表 示 列 名 ，Type 表示 数据 类 型 。Key 和 Extra 会 在 6.8 节 进 行 介 绍 ， 这 里 就 不 多 讲 
了 。 另 外 ， 数 据 类 型 INT 后 面 紧 跟 着 的 数值 是 MySQL 自动 设置 的 ， 表 示 位 数 。 




















特殊 的 SHOW 


使 用 SHOW 命令 能 够 显示 数据 库 名 ( 一 4.2 节 )、 表 名 (一 4.5 节 )、 表 的 结构 以 及 字符 编码 设置 (一 3.3.4 
节 ) 等 各 种 各 样 的 信息 。 

然而 ，SHOW 是 其 他 RDBMS 的 SQL 中 所 没有 的 ， 它 是 MySQL 特有 的 命令 "。 因 此 ， 一 些 精通 SQL 的 人 
认为 应 该 尽量 不 去 使 用 只 能 在 MySQL 中 使 用 的 命令 。 但 是 ， 本 书 是 介绍 MySQL 的 书 ， 而 且 SHOW 也 是 一 个 
非常 方便 的 命令 ， 所 以 本 书 大 量 使 用 了 该 命令 。 此 外 ，4.3.1 节 提 到 本 书 统一 使 用 小 写字 母 表示 不 需要 输入 “;” 
的 命令 ，sHOW 虽然 是 MySQL 特有 的 命令 ,但 是 使 用 时 需要 在 后 面 加 上 “;”， 因 此 ， 在 本 书 中 该 命令 和 普通 的 
SQL 命令 一 样 使 用 大 写字 母 表示 。 

























































































































































































4.7 ”向 表 中 插入 数据 


虽然 表 tb1 创建 成 功 了 ， 但 是 最 重要 的 数据 还 没有 搬入 进去 ， 所 以 下 面 我 们 试 着 添加 数据 。 




















4.7.1 插入 数据 


我 们 可 以 使 用 下 面 的 命令 向 表 中 插入 数据 。 
向 表 中 插入 数据 








在 VALUES 后 面 的 () 中 ,需要 按照 列 的 顺序 用 “, ”来 区 分 各 数据 。 这 里 我 们 试 着 向 DD 公司 
员工 信息 表 tpl 中 插入 表 4-2 的 数据 。 


表 4-2 向 表 tb1 中 插入 的 数据 


























列 empid 中 插入 的 值 列 name 中 插入 的 值 列 age 中 插入 的 值 
A101 佐 且 40 
A102 高 桥 28 
A103 中 川 20 
A104 渡 边 23 
A105 泽 35 









































中， 12c Oracle 数据 库 中 也 有 SHOW 命令 ， 例 如 SHOW PDBS 等 。 一 一 译 者 注 
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例如 第 1 行 数据 表示 员工 号 为 A101、 名 字 为 伍 芯 的 员工 ， 其 年 龄 是 40 岁 。 首 先 创建 5 条 (5 
个 人 的 ) 数据 ， 一 个 人 的 数据 就 是 一 条 记录 (一 1.1.2 节 )。 

另外 ， 因 为 列 name 被 设置 成 了 VARCHAR (10) ， 所 以 我 们 无 法 输入 多 于 10 个 字符 的 数据 。 但 
是 在 MySQL 中 ， 即 使 输入 了 多 于 指定 字符 数 ( 这 里 是 10 个 字符 ) 的 数据 也 不 会 报错 ， 而 是 会 忽略 
无 法 插入 的 字符 ( 这 里 是 指 10 个 字符 之 后 的 数据 ) "， 这 一 点 需要 注意 。 

数值 数据 可 以 直接 插入 ， 字 符 串 数据 需要 用 " "或 者 ' “' 括 起 来 。 本 书 使 用 ' ，' 将 字符 串 数 
据 括 起 来 (一 4.4.3 节 )。 

这 次 也 会 输入 中 文 。 我 们 可 以 通过 在 命令 提示 符 中 启动 中 文 输入 法 来 输入 。 注 意 , 一 定 要 使 用 
半角 字符 输入 中 文 以 外 的 内 容 。 




















P 执行 内 容 
> 执行 前 ( 表 tb1 ) 











empid name age 


D4 















































> 执行 后 

empid name age 

A101 佐 膝 40 一 一 一 一 一 插入 第 1 条 记录 
操作 方法 
































通过 下 面 的 命令 ， 向 表 tb1 的 列 empid、 列 name 和 列 age 中 分 别 输入 'A101'、' 佐 蕨 '、40 








| INSERT INTO tbl VALUES('A101',' 佐 芯 ' ,40); 


执行 结果 





mysql> INSERT INTO tbl VALUES ('R101'，' 佐 藤 ' ,40); 
Query OK, 1 row affected (0.12 sec) 




















执行 成 功 的 话 会 显示 Query OK。 如 果 显 示 You have an error, 请 确认 拼写 是 否 正确 
注意 ， 不 能 出 现 全 角 空 格 。 

这 样 就 插入 了 第 1 条 记录 。 第 2 条 记录 也 可 以 用 同样 的 方法 插入 ， 不 过 我 们 可 以 用 其 他 更 简单 
的 方法 来 操作 。 请 试 着 按 1 键 或 了 键 ， 这 样 就 会 显示 出 之 前 输入 的 命令 历史 。 























四 该 动作 主要 受到 MySQL 的 SQL Modes 的 影响 ， 例 如 当 SQL Modes 设 为 STRICT_TRRNS_TRBLES 时 会 报 出 
Data too long for column 'XXX' at row XX 的 错误 。 一 一 译 者 注 


52 | 第 4 章 创建 数据 库 


4.7.2 向 表 tb1 中 添加 第 2 条 记录 


一 一 一 一 一 一 一 一 一 


下 面 我 们 使 用 上 下 键 更 高 效 地 输入 第 2 条 记录 。 














区 执行 内 容 


> 执行 前 ( 表 tb1 ) 





















































empid name age 
A101 佐 艾 40 
> 执行 后 
empid name age 
A101 佐 艾 40 
A102 高 桥 28 @ 一 一 一 一 一 插入 第 2 条 记录 
操作 方法 











人 按键 或 上 | 键 ， 会 显示 出 输入 第 1 条 记录 时 使 
(全 

C) 通过 *- 键 将 光标 移动 到 需要 修改 的 字符 处 。 

(3) 使 用 Delete 键 或 者 Back space 键 按照 下 面 的 内 容 进 行 修改 ， 然 后 按 Enter 键 。 











的 命令 “INSERT INTO tbl VALUES('A1l101',' 






























































和 命令 提示 符 一 样 ， 通 过 1 键 或 上 键 能 够 显示 出 以 前 输入 的 命令 历史 。 如 果 有 需要 修改 的 地 
方 ， 可 以 使 用 一 键 或 一 键 移动 光标 ， 编 辑 需 要 修改 的 内 容 。 无 论 光 标 在 这 一 行 的 哪个 位 置 ， 我 们 只 
要 按 下 Enter 键 都 可 以 执行 该 命令 。 另 外 ,， 按 F7 键 可 以 让 历史 命令 列表 显示 出 来 。 往 后 会 输入 更 复 
杂 的 SQL 语句 ， 记 下 来 这 些 技巧 会 方便 不 少 。 























如 果 执 行 结果 显示 Query OK，1 row affected (0.00 sec) ， 就 表示 第 2 条 记录 插入 成 
功 了 。 然 后 使 用 同样 的 方法 插入 第 3 条 记录 。 





操作 方法 























通过 下 面 的 命令 插入 第 3 条 记录 。 


;INSERT INTO tbl VALUES('A103',' 中 川 ' ,20); 
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4.7.3 ”指定 列 名 插入 记录 


CR 
我 们 试 着 用 其 他 方法 插入 第 4 条 记录 。 上 面 的 方法 需要 按照 设置 好 的 列 的 顺序 输入 数据 ， 但 其 
实 我 们 也 可 以 忽略 列 的 顺序 ， 通 过 指定 列 名 来 插入 记录 。 
指定 列 名 插入 记录 


; INSERT INTO 表 名 ( 列 名 1, 列 名 2...) VALUES ( 数据 1, 数据 2.…) ; 











下 面 就 来 使 用 这 种 方法 插入 第 4 条 记录 。 这 里 我 们 故意 改变 列 的 顺序 插入 记录 。 





P 执行 内 容 
> 执行 前 ( 表 tb1 ) 
















































































empid name age 
A101 佐 且 40 
A102 高 村 28 
A103 中 川 20 
4 
上 执行 后 
empid name age 
A101 佐 芯 40 
A102 高 村 28 
A103 中 川 20 
A104 渡 边 2 @ 一 一 一 一 插入 第 4 条 记录 
操作 方法 








输入 下 面 的 命令 ， 插 入 第 4 条 记录 。 




















请 使 用 上 面 介绍 的 任意 一 种 方法 插入 第 5 条 记录 。 


4.7.4 一 次 性 输入 记录 


OO 


有 一 种 方法 可 以 一 次 性 输入 多 行 记录 。 
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向 表 中 插入 多 行 记录 


: INSERT INTO 表 名 ( 列 名 1, 列 名 2...) VALUES (数据 1, 数据 2.…) ，( 数据 1, 数据 2...),( 数据 1, 数据 
: 全 


对 于 4.7.1 节 中 的 示例 ， 我 们 可 以 使 用 下 面 的 方法 插入 记录 。 

















INSERT INTO tbl (empid,name,age) VALUES ('A101',' 佐 有 蓉 ' ,40),('A102',' 高 桥 ' ,28)， 


(be 





虽然 较 长 的 SQL 语句 也 可 以 在 中 间 换 行 输入 ,但 是 VALUES 等 关键 字 如 果 不 在 一 行 ， 就 会 发 
生 错 误 。 此 外 ， 数 据 中 间 也 不 能 换行 。 


4.8 显示 数据 


至 此 我 们 就 完成 了 数据 的 输入 ， 下 面 我 们 试 着 把 输入 的 数据 显示 出 来 。 


6 4.8.1 显示 数据 


使 用 SELECT 命令 能 够 让 列 的 数据 显示 出 来 。SELECT 是 SQL 语句 中 使 用 频率 最 高 的 命令 。 
显示 各 列 的 数据 


: SELECT 列 名 1， 列 名 2... FROM 表 名 ; 





例如 ， 当 显示 表 tbl 的 列 empid 和 列 name 时 ， 需 要 使 用 下 面 的 命令 。 


SELECT empid,name FROM tbl; 


一 个 一 个 地 指定 列 名 有 些 麻 烦 ， 这 时 使 用 表示 所 有 列 的 “*”( 星 号 ) 会 方便 许多 。 
此 外 ， 从 第 8 章 开 始 我 们 将 会 学 习 如 何 才 能 只 显示 满足 条 件 的 记录 。 在 此 之 前 ， 我 们 暂时 使 用 
“SELECT * FROM 表 名 ;” 来 显示 所 有 记录 。 


P 执行 内 容 
> 执行 后 ( 表 tb1 的 记录 ) 


empid name age 
A101 佐 且 40 
A102 高 桥 28 



































邮 
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( 续 ) 
empid name age 
A103 中 川 20 
A104 渡 边 23 
A105 西 泽 35 
操作 方法 








输入 下 面 的 命令 ， 显 示 所 有 的 记录 。 


























mysql> SELECT * FROM tbl; 






































天 Pe 下 + 
empid name age 
+------- 机 = +------ 十 
RAR101 佐 蔷 40 
RAR102 高 村 28 
A103 中 川 20 
A1l04 渡 边 pe: 
A1l05 西 泽 35 
人 和 十 


5 rows in set (0.00 sec) 








P 使 用 SELECT 输出 指定 的 值 


SELECT 命令 还 能 用 于 显示 与 数据 库 无 关 的 值 。 例 如 输入 


select :测试 '; 





字符 串 “ 测 试 ”就 会 像 下 面 这 样 显示 出 来 。 














mysql> select ' 测 试 '; 





1 row in set (0.00 sec) 
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这 种 方法 适用 于 确认 函数 的 值 或 计算 结果 。 例 如 输入 “SELECT (2+3) *4;” 后 按 Enter 键 ， 
就 会 显示 出 “20”。 

另外 ，SELECT 在 MySQL 以 外 的 RDBMS 中 也 能 使 用 。 无 论 对 多 么 重要 的 表 执 行 该 命令 都 不 
会 破坏 表 中 数据 ， 所 以 大 家 可 以 放心 练习 。 











4.9 (准备 ) 复制 表 tb1 


第 5 章 以 后 会 使 用 和 表 tbl 内 容 相同 的 表 tbl1A~ tb1K 来 介绍 MySQL 的 表 操作 。 当 然 ， 使 用 之 
前 介绍 的 CREATE TABLE 命令 也 可 以 创建 这 些 表 ， 不 过 会 麻烦 一 些 。 
下 面 我 们 通过 复制 表 tbl 来 创建 11 个 表 。 表 的 复制 方法 会 在 7.1 节 详 细 介绍 ， 这 里 只 要 在 MySQL 
监视 器 中 执行 下 面 的 命令 即 可 。 第 2 行 之 后 可 以 使 用 了 键 ， 仅 修改 表 名 即 可 ， 这 样 很 快 就 能 完成 。 












































CREATE TABLE tblA SELECT * FROM tbl; 
CREATE TABLE tblB SELECT * FROM tb]l; 


CREATE TABLE tblK SELECT * FROM tb]l; 











我 们 可 以 通过 SHOW TABLES 命令 确认 是 否 成 功 创建 了 表 。 

















mysql> CREATE TABLE tblA SELECT * FROM tbl: 
Query OK, 5 rows affected (0.20 sec) 
Records: 5 Duplicates: 0 Warnings: 0 


mysql> CREATE TABLE tblB SELECT * FROM tbl: 
Query OK, 5 rows affected (0.09 sec) 
Records: 5 Duplicates: 0 Warnings: 0 
mysql> CREATE TABLE tblK SELECT * FROM tbl; 
Query OK, 5 rows affected (0.19 sec) 


Records: 5 Duplicates: 0 Warnings: 0 


mysql> SHOW TABLES; 


12 rows in set (0.00 sec) 











4.10 和 总结 


本 章 介绍 了 以 下 内 容 。 


@ 创 建 数据 库 的 方法 
@ 创 建 表 的 方法 

@ 显 示 列 结构 的 方法 
@ 插 入 数据 的 方法 

@ 显 示 所 有 数据 的 方法 


SQL 的 初学 者 要 先 反复 练习 基本 的 SQL 语句 ， 达 到 牢记 于 心 的 程度 。 另 外 ， 请 灵活 使 用 附录 3 
中 的 练习 。 





> 自我 检查 
下 面 检查 一 下 本 章 学 习 的 内 容 是 否 全 部 理解 并 掌握 了 。 


口 能 够 理解 “CREATE DATABASE...” 的 使 用 方法 
口 能 够 理解 “CREATE TABLE...” 的 使 用 方法 
口 能 够 理解 “DESC...” 的 使 用 方法 

口 能 够 理解 “INSERT INTO...” 的 使 用 方法 
口 能 够 理解 “SELECT * FROM...” 的 使 用 方法 





i 练习 题 





问题 1 


使 用 1 行 命令 显示 2 次 表 tb1 的 所 有 数据 。 
































人 | 只 显示 2 次 结果 ， 但 命令 是 1 行 。 








问题 2 




















数据 库 mysql 的 表 user 的 列 user 中 包含 了 用 户 信息 ， 请 显示 这 个 信息 。 


HINT 上 SE 
必须 用 root 权限 执行 。 
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局 参考 答案 


















































问题 1 

执行 下 面 的 命令 。 

执行 结果 

mysql> SELECT * FROM tbl; SELECT * FROM tbl: 

2 ns 人 十 
empid name age 

Ee 开本 二 全 二 二 十 
A1l01 佐 芯 40 
A1l02 高 村 28 
RA103 中 川 20 
A104 渡 边 23 
RA105 西 泽 35 

各 Ss 和 十 
































+------- +------ +------ + 
empid name age 
Fes ee 和 + 
A1l01 佐藤 40 
RA102 高 村 28 
RA103 中 川 20 
A104 渡 边 23 
A1l05 西 泽 35 
+------- 二 +------ + 


5 rows in set (0.00 sec) 

















use mysql 
ey 
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mysql> use mysql 





Database changed 
mysql> SELECT user FROM user; 








4 rows in set (0.00 sec) 





执行 后 使 用 use dbl 选择 原来 的 数据 库 。 
在 数据 库 dbl 中 执行 下 面 的 命令 也 会 得 到 相同 的 结果 。 


SELECT user FROM mysql .user; 


MySQL 的 文档 
本 书 使 用 的 MAMP 附属 的 MySOL 是 5.6 版 本 ， 我 们 可 以 通过 下 面 的 URL 查看 它 的 参考 手册 。 







































































© MySQL 5.6 参考 手册 
https://dev.mysql.com/doc/refman/5.6/en/ 








当前 ( 2018 年 5 月 ) MySQL 的 最 新 版 本 是 8.0， 我 们 可 以 通过 下 面 的 URL 查看 它 的 参考 手册 。 
MySQOL 8.0 参考 手册 
https://dev.mysql.com/doc/refman/8.0/en/ 
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数据 库 中 一 旦 设置 了 数据 类 型 ， 之 后 再 修改 就 会 很 麻烦 。 本 
我 们 需要 牢 牢 记 住 可 以 在 MySOL 中 设置 的 具有 代表 性 的 数据 
类 型 以 及 各 类 型 数据 的 输入 方法 。 




























































































5.1 什么 是 数据 类 型 


如 图 5-1 所 示 ， 数 据 库 的 表 中 只 能 输入 各 个 列 指定 格式 的 数据 。 例 如 ， 指 定 为 “数值 类 型 ”的 
列 中 不 能 输入 字符 等 数据 。 指 定 为 “日 期 类 型 ”的 列 中 ， 只 能 输入 日 期 数据 。 这 种 数据 的 格式 称 为 
数据 类 型 。 














数据 类 型 一 区 | VARCHAR(5) 类 型 INT 类 型 DATE 类 型 
























































输入 大 小 的 
限制 



































飞 岛 二郎 456789 2014-8-1 


一 列 输入 相 
种 类 的 数据 














串 吕 
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数据 类 型 具有 降低 输入 错误 数据 的 可 能 性 、 保 证 数据 库 整 体 可 信赖 性 的 特点 。 

但 是 MySQL 和 其 他 RDBMS 相 比 ， 数 据 输入 的 检查 功能 相对 趋 于 宽松 。 因 此 ， 在 很 多 情况 下 
即使 输入 的 数据 不 符合 设置 的 数据 类 型 也 不 会 发 生 错误 。 例 如 向 数值 类 型 的 列 中 输入 文字 就 不 会 发 
生 错 误 ， 而 是 会 输入 “0”。 这 就 导致 输入 的 数据 格式 和 用 户 想 要 的 数据 格式 出 现 偏 差 。 大 家 需要 注 
意 这 种 情况 。 




















5.2 ”数值 类 型 


数值 类 型 的 列 中 需要 输入 数值 。 


5.2.1 数值 类 型 的 种 类 
表 5-1 中 列 出 了 常用 数值 型 数据 类 型 的 种 类 。 本 书 的 示例 省 略 了 位 数 的 指定 。 


表 5-1 常用 的 数值 型 数据 类 型 



























































数据 类 型 含义 对 应 的 范围 

INT - -2 147 483 648 ~ 2 147 483 647 

TINYINT 极 小 的 整数 -128 ~ 127 

SMALLINT | 小 整数 -32 768 ~ 32 767 

MEDIUMINT | 中 等 整数 -8 388 608 ~ 8 388 607 

BIGINT 大 整数 -9 223 372 036 854 775 808 ~ 9 223 372 036 854 775 807 





-3.402 823 466E+38 ~ -1.175 494 351E-38 
FLOAT 单 精度 浮 点 数 0 
1.175 494 351E-38 ~ 3.402 823 466E+38 


—1.797 693 134 862 315 7E+308 ~ -2.225 073 858 507 201 4E-308 
DOUBLE 双 精 度 浮 点 数 0 

2.225 073 858 507 201 4E-308 ~ 1.797 693 134 862 315 7E+308 
DECIMAL ( 最 大 位 数 ， 小 数 点 之 后 的 位 数 ) 的 格式 中 的 “最 大 位 数 ” 可 
DECIMAL “| 精确 小 数 以 指定 不 大 于 65 的 值 ,“ 小 数 点 之 后 的 位 数 ” 可 以 指定 不 大 于 30 的 值 。 
不 会 产生 误差 
























































虽然 还 有 很 多 其 他 的 数值 类 型 ， 但 本 书 的 整数 数据 只 使 用 INT 类 型 ， 包 含 小 数 点 的 数据 只 使 
用 DOUBLE 类 型 。 
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人 5.2.2 输入 数值 数据 

输入 数据 时 可 以 加 上 正 负 号 “+”“-”。 

数值 也 可 以 使 用 指数 表示 法 输入 。 在 使 用 指数 表示 法 输入 的 情况 下 ， 需 要 使 用 符号 “E”。 
“O E+ 人 ”表示 “OO 〇 乘 以 10 的 人 次 方 ”。 例 如 ，6.02 x 103” 可 以 表示 为 “6.02E+23”。 

举例 来 说 ， 当 向 表 tblA 的 INT 类 型 的 列 age 中 输入 10000 (1 万 ) 时， 如 果 使 用 指数 格式 ， 就 
是 下 面 这 种 形式 。 


INSERT INTO tblA (age) VALUES (1E+4); 


另外 ， 在 以 后 介绍 的 内 容 中 ， 我 们 将 使 用 和 表 tbl 内 容 相 同 的 表 tb1A ~ tblK (一 4.9 节 ) 进行 
练习 。 还 没有 创建 这 些 表 的 读者 ， 也 可 以 使 用 下 载 文件 table 文件 夹 中 的 tblA_K_make.txt 文件 创建 
这 些 表 。( 请 参考 前 面 介绍 的 下 载 示例 。 ) 
































5.3 ”字符 串 类 型 


S 5.3.1 字符 串 类 型 的 种 类 


字符 串 类 型 主要 包括 表 5-2 中 列 出 的 几 类 。 





表 5-2 ”常用 的 字符 串 数 据 类 型 



















































































数据 类 型 含义 对 应 的 范围 
CHAR 国定 长 度 字 符 串 长 度 不 超过 255 个 字符 
VARCHAR 可 变 长 度 字 符 串 1~ 65 532 字 节 。 字 符 数 的 上 限 取决 于 使 用 的 字符 编码 
TEXT 长 文本 字符 串 长 度 不 超过 65 535 个 字符 
LONGTEXT 极 长 的 文本 字符 串 长 度 不 超过 4 294 967 295 个 字符 
现在 只 要 记 住 长 度 不 超过 255 个 字符 的 是 VARCHAR， 超 过 255 个 字符 的 是 TEXT 就 足够 了 。 











VARCHAR 和 CHAR 能 够 在 () 中 指定 位 数 。 例 如 VARCHAR (100) 表示 可 以 存储 不 超过 100 个 字符 
的 字符 串 。 
CHAR 类 型 为 固定 长 度 的 字符 串 ， 在 保存 数据 的 时 候 ， 字 符 数 如 果 没 有 达到 () 中 指定 的 数量 
就 会 用 空格 填充 。 但 是 ， 读 取 时 这 些 填 充 的 空格 会 被 自动 删除 2?。 而 VARCHAR 为 可 变 长 度 字符 串 ， 
保存 数据 时 不 会 填充 空格 。 


中 Oracle 等 数据 库 不 会 自动 删 掉 填充 的 空格 。 

















译 者 注 
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本 书 使 用 的 字符 串 类 型 是 VRACHAR。 


5.3.2 ”输入 字符 串 


字符 串 数据 需要 用 双 引 号 “"” 或 者 单 引号 “'” 括 起 来 。 如 果 输 入 的 字符 串 数据 中 包含 “'”， 
那么 “'” 之 后 出 现 的 “' ”就 会 被 解释 为 表示 字符 串 结 束 的 “'”。 

所 以 ， 当 把 ' ' 当成 字符 输入 时 ， 要 在 它 的 前 面 加 上 “\”"， 该 操作 称 为 转 义 人 处理 (escape 
processing )。 


下 面 是 向 表 tb1B 的 列 name 中 输入 带 “'” 的 ' 西 泽 ' 的 例子 。 





INSERT INTO tblB (name) VALUES ('\' 西 泽 \''); 





同样 ， 当 输入 “\” 时 ， 也 需要 在 前 面 加 上 “\”, 输入 “\\”。 


忆 53.3_VARCHAR 和 CHAR 的 位 数 尘 位 


在 MySQL 4.1 以 后 的 版 本 中 ，VARCHAR 和 CHAR 的 () 中 指定 的 位 数 单位 变 成 了 “字符 ”"。 以 
VARCHAR (10) 为 例 ， 不 管 输入 的 是 中 文 还 是 半角 英文 字母 和 数字 ， 最 多 都 只 能 保存 10 个 字符 。 
而 在 4.0 之 前 的 版 本 中 ， 位 数 单位 是 “ 字 节 ”， 请 注意 版 本 间 的 差别 。 














5.4 日 期 与 时 间 类 型 


能 够 保存 日 期 或 时 间 的 列 的 数据 类 型 包括 DATE (日 期 )、TIME ( 时间 )、YERAR (年 )， 以 及 把 
日 期 和 时 间 组 合 在 一 起 的 DATETIME 等 ( 见 表 5-3 )。 





表 5-3 常用 的 日 期 与 时 间 数 据 类 型 




















数据 类 型 信义 对 应 的 范围 
DATETIME 期 和 时 间 1000-01-01 00:00:00 ~ 9999-12-31 23:59:59 
DATE 期 1000-01-01 ~ 9999-12-31 





























1901 ~ 2155 (4 位 时 ) 
1970 ~ 2069 ( 70 ~ 69 ) (2 位 时 ) 


TIME 时 间 -838:59:59 ~ 838:59:59 


YEAR 午 
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DATETIME 类 型 可 以 处 理 日 期 和 时 间 的 数据 。 这 里 大 家 只 要 记 住 可 以 一 起 保存 日 期 和 时 间 的 
DATETIME 类 型 ， 以 及 只 能 保存 日 期 的 DATE 类 型 就 足够 了 。 




















5.4.2 输入 日 期 与 时 间 类 型 的 数据 


过 一 一 一 全 一 一 人 一 一 一 一 一 一 一 一 
日 期 与 时 间 类 型 的 数据 需要 使 用 单 引 号 “' ”或 者 双 引 号 “" ” 括 起 来 。 在 MySQL 中 ， 日 期 必 
须 以 YYYYMM-DD 的 格式 输入 ， 时 间 必 须 以 HH:MM:SS 的 格式 输入 。 
我 们 来 练习 一 下 日 期 数据 的 输入 方法 。 下 面 创建 列 a 为 日 期 类 型 的 表 t_date， 并 输入 值 为 
“2018-5-3” 的 日 期 数据 。 












































> 执行 内 容 


> 表 t_date 的 结构 
列 a 


数据 类 型 DATE 


D4 


向 列 a 中 输入 内 容 为 “2018-5-3” 的 日 期 数据 


» 























a 


2018-05-03 














操作 方法 





由 输入 下 面 的 命令 ， 创 建 表 t_date。 









































mysql> CREATE TABLE t date (a DATE); 
Query OK, 0 rows affected (0.20 sec) 
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mysql> INSERT INTO t date (a) VALUES('2018-5-3'); 
Query OK, 1 row affected (0.06 sec) 


mysql> SELECT * FROM t date; 


+------------ 十 
| a | 
+------------ 十 
| 2018-05-03 | 
惠 守 记 守 3 十 





1 row in set (0.00 sec) 








修改 提示 符 的 字符 串 

在 MySQL 监视 器 运行 的 时 候 ， 我 们 能 够 在 提示 符 “>” 中 指定 自己 想 显 示 的 内 容 。 

当 MySOQL 监视 器 启动 时 (一 3.4.3 节 )， 我们 可 以 指定 “--prompt = 提示 符 内 容 ” 选 项 设置 提示 符 ， 但 
如 果 设 置 的 提示 符 中 包含 中 文 就 会 发 生 乱 码 ( 在 Windows 的 情况 下 )。 当 发 生 这 种 情况 时 ， 我 们 可 以 在 MySQL 
监视 器 启动 后 使 用 prompt 命令 修改 提示 符 的 内 容 。 













































































国 设 置 作为 提示 符 显 示 的 文本 


eeeeeeeeeseeeeeeeeeeeeeoeeoeeoeoeeeoeoooooeeeoeoeooeeeooeoeoeeoeeeeoeoooeeooeeeooeoeoooeoeooeoeoeeoooeoeoeooeoeoeoooeoeoeoeoeeeoeooooeoeoeoeoeoeoeeoooeeeeeeeeoeseo。 


另外 ， 我 们 还 可 以 通过 “\da'” 设置 数据 库 名 ， 通 过 “\h ”设置 主机 名 ， 通 过 “\u” 设置 用 户 名 。 下 面 是 把 
提示 符 设置 成 “我 的 提示 可 能 会 很 好 + 数据 库 名 +>” 的 示例 。 


prompt 我 的 提示 可 能 会 很 好 \d> 


执行 上 面 的 命令 会 得 到 以 下 结果 。 
































mysql> prompt 我 的 提示 可 能 会 很 好 \d> 
PROMPT set to ' 我 的 提示 可 能 会 很 好 \d>' 
我 的 提示 可 能 会 很 好 db1 > 























执行 下 面 的 命令 ， 可 以 回 到 初期 的 状态 。 


prompt mysql> 
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本 章 介 绍 了 以 下 内 容 。 

@ 数 据 类 型 是 什么 

@ 能 够 设置 的 数据 类 型 的 种 类 
@ 如 何 设置 列 的 数据 类 型 





在 实际 操作 中 ，VARCHAR 应 该 设置 成 多 少 个 字符 、 该 使 用 INT 还 是 TINYINT 等 都 是 非常 重 
要 的 问题 。 我 们 要 记 住 这 些 具有 代表 性 的 数据 类 型 。 








> 自我 检查 
下 面 检查 一 下 本 章 学 习 的 内 容 是 否 全 部 理解 并 掌握 了 。 




















口 能 够 掌握 主要 数据 类 型 的 种 类 和 含义 

口 能 够 创建 列 为 INT 或 DOUBLE 等 数值 类 型 的 表 

口 能 够 创建 列 为 VARCHRAR 或 CHAR 等 字符 串 类 型 的 表 

口 能 够 创建 列 为 DATETIME 或 YEAR 等 日 期 与 时 间 类 型 的 表 





二 练习 题 





问题 1 




















创建 表 t_time， 并 让 该 表 拥 有 能 够 存储 “年 月 日 时 分 秒 ” 的 列 col_time。 另 外 ， 请 向 列 col_time 中 插 
六 20 SEE IO Ss ll 让 六 汪 

















基本 方法 是 设置 DATETIME 类 型 的 列 并 输入 数据 。 





问题 2 


设置 MySQL 监视 器 提示 符 的 显示 内 容 为 “当前 数据 库 名 : 




















主机 名 > 。 























HINT 
4 请 参考 前 面 修改 MySQL 监视 器 提示 符 的 示例 。 试 着 设置 成 方便 自己 使 用 的 提示 符 。 












































ta 参考 答案 














问题 1 
执行 下 面 的 命令 。 


CREATE TABLE t time (col time DATETIME); 


INSERT INTO t time VALUES('2018-11-17 20:35:15'); 





mysql>CREATE TABLE t time (col time DRATETIME) 
Query OK, 0 rows affected (0.09 sec) 





mysql>INSERT INTO t time VALUES('2018-11-17 20:35:15'); 
Query OK, 1 row affected (0.06 sec) 


mysql>SELECT * FROM t time; 


于 时 + 
| col time | 
i + 
| 2018-11-17 20:35:15 | 
和 + 


1 row in set (0.00 sec) 

















问题 2 
输入 下 面 的 命令 。 


prompt \d:\h> 


mysql>prompt \d:\h> 
PROMPT set to '\d:\h>' 
dbl:localhost> 














第 6 章 ”修改 表 


下 面 对 在 第 4 章 创 建 的 员工 信息 表 进 行 修改 。 本 章 将 介绍 如 
何 修改 列 的 数据 类 型 。 

另外 ， 本 章 将 会 使 用 和 表 tb1 内 容 相 同 的 表 tb1C、 表 tb1D 
等 进行 操作 (一 4.9 节 )。 如 果 使 用 表 tb1 进行 操作 ， 请 忽略 中 
间 修 改 的 内 容 ， 继 续 往 下 阅读 。 





























































































































6.1 修改 表 的 列 结构 


在 第 4 章 中 ,我 们 创建 出 指定 了 列 名 和 数据 类 型 的 表 (一 4.4.3 节 )， 并 且 依 照 各 列 的 数据 类 型 
插入 了 相应 的 数据 (一 4.7 节 )。 

即使 把 表 创 建 了 出 来 ， 在 实际 操作 中 也 需要 经 常 对 表 进 行 修改 。 比 如 希望 可 以 输入 更 多 的 字 
符 、 需 要 添加 列 、 想 要 修改 列 名 等 情况 。 

















我 们 可 以 使 用 ALTER TABLE 命令 修改 列 的 结构 。 根 据 需要 修改 的 类 型 ， 可 以 像 下 面 这 样 使 用 
带 有 MODIFY、ADD、CHANGE、DROP 的 语句 。 





























3 当 修改 列 的 定义 时 : ALTER TABLE ... MODIFY ... 
2 当 添加 列 时 : ALTER TABLE ... RDD ... 

2 当 修改 列 名 和 定义 时 : ALTER TABLE ... CHANGE ... 
2 当 删 除 列 时 : ALTER TABLE ... DROP ... 


首先 来 看 一 下 如 何 自由 地 修改 列 结构 。 下 面 会 介绍 修改 数据 类 型 、 添 加 列 、 修 改 列 的 位 置 、 修 
改 列 名 、 删 除 列 这 一 系列 的 操作 。 
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6.2 ”修改 列 的 数据 类 型 


任何 时 候 都 可 以 修改 列 的 数据 类 型 。 例 如 当 我 们 需要 向 设置 成 VARCHAR 的 列 中 输入 大 量 字符 
时 ， 就 可 以 把 列 修改 为 TEXT 类 型 。 

但 是 ， 数 据 类 型 的 修改 必须 具有 兼容 性 。 不 具有 兼容 性 的 修改 会 导致 错误 发 生 。 需 要 注意 的 是 ， 
即使 数据 类 型 可 以 修改 ， 也 可 能 会 发 生 数 据 变 成 没有 意义 的 值 、 全 部 或 者 部 分 数据 丢失 之 类 的 情况 。 如 
果 把 已 经 输入 了 100 个 字符 的 列 修改 为 VARCHAR (50) ， 训 无 疑问 ， 第 50 个 字符 之 后 的 数据 就 会 丢失 。 

当 修 改 列 的 数据 类 型 时 ， 我 们 需要 使 用 下 面 的 命令 。 


修改 列 的 数据 类 型 
























































首先 准备 一 个 内 容 和 员工 信息 表 tbl 完全 相同 的 表 tb1C。 当 然 也 可 以 使 用 表 tbl 进行 操作 。 
试 着 修改 表 tb1C 的 列 结构 。 列 name 中 允许 输入 的 字符 数 原本 不 超过 10 个 ， 现 在 我 们 把 它 修 
改 为 不 超过 100 个 。 
P 执行 内 容 
戎 (执行 前 ) 表 tb1C 的 列 结构 








列 empid name age 




















数据 类 型 VARCHAR (10) VARCHAR (10) INT 
、 4 | 把 这 个 数字 改 为 100 


PP (执行 后 ) 把 列 name 修改 为 VARCHAR(100) 









































列 empid name age 
数据 类 型 VARCHAR (10) VARCHAR (100) INT 
操作 方法 




















执行 下 面 的 命令 。 























mysql> ALTER TABLE tbl1C MODIEY name VARCHAR(100); 
Query OK, 5 rows affected (0.19 sec) 
Records: 5 Duplicates: 0 Warnings: 0 


mysql> DESC tblC: 

















+----- +--------- +------- 十 
| Field | Type | Null | Key | Default | Extra | 
+------- +-------------- +------ +----- +--------- +------- 十 
| empid | varchar (10) | YES | | NU | 

| name | varchar(100) | YES | | NU | 

| age | int (11) | YES | | NU | | 
+------- +-------------- +------ +----- +--------- +------- + 


3 rows in set (0.02 sec) 








表示 姓名 的 列 name 显示 为 VARCHAR (100) 。 这 样 就 修改 完成 了 。 





修改 数据 类 型 要 慎重 


在 大 多 数 的 情况 下 ， 存 储 了 “开头 不 是 0 且 仅 由 数值 字符 组 成 的 数据 ”的 列 能 按照 “INT 类 型 ”一 
“VARCHAR 类 型 ”一 “INT 类 型 ”的 方式 进行 修改 。 但 是 ， 如 果 列 中 存在 数据 ， 原 则 上 就 不 应 该 再 修改 列 的 数 
据 类 型 了 。 
















































































6.3 ”添加 列 


现在 员工 信息 表 tb1l 中 定义 了 empid、name 和 age 这 3 个 列 。 下 面试 着 添加 能 够 输入 员工 出 生 
日 期 的 DATETIME 类 型 的 列 birth。 
我 们 可 以 使 用 下 面 的 命令 将 新 建 的 列 添加 到 最 后 的 位 置 。 


将 新 建 的 列 添加 到 最 后 的 位 置 
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PP 执行 内 容 
> 执行 前 ( 表 tb1 的 列 结构 ) 


列 empid name age 








数据 类 型 VARCHAR (10) VARCHAR (100) | INT 


D4 



























































> 执行 后 
列 empid name age birth 
数据 类 型 VARCHAR (10) VARCHAR (100) | INT DATETIME 
LE 添加 该 列 
探 必 测 法 





() 执行 下 面 的 命令 。 









































mysql> ALTER TABLE tblC ADD birth DRATETIME : 
Query OK, 0 rows affected (0.33 sec) 
Records: 0 Duplicates: 0 Warnings: 0 


mysql> DESC tblC: 














+------- +-------------- +------ +----- +--------- +------- + 
| Field | Type | Null | Key | Default | Extra | 
+------- +-------------- +------ +----- +--------- +------- 十 
| empiqd | varchar(10) | YES | | NULL | 
| name | varchar(100) | YES | | NUL | 
| age | int (11) | YES | | NUL | | 
| birth | datetime | YES | | NULL | | 
+------- +-------------- +------ +----- +--------- +------- 十 
) 























上 面 的 执行 结果 显示 了 员工 信息 表 tb1C 的 列 结构 。 至 此 ， 我 们 就 成 功 添加 了 名 为 birth 且 允 六 
输入 日 期 和 时 间 的 列 。 























在 输入 出 生日 期 的 情况 下 ， 使 用 仅 能 输入 日 期 的 DATE 类 型 就 已 经 足够 了 。 但 这 次 我 们 选择 更 
进一步 ， 使 列 birth 中 能 够 输入 包括 出 生 时 间 在 内 的 出 生日 期 。 





我 们 试 着 向 表 tb1C 中 添加 包括 出 生日 期 在 内 的 员工 信息 记录 。 具 体 来 说 ， 就 是 向 表 tb1C 中 插 
入 empid 为 “N111”、name 为 “松田 ”、age 为 “38”、birth 为 “1980-11-10” 的 记录 。 插 入 数据 后 ， 
试 着 把 所 有 的 记录 都 显示 出 来 。 








PP 执行 内 容 
PP (执行 前 ) 表 tb1C 


























empid name age birth 

A101 佐 茧 40 ULL 
A102 高 桥 28 ULL 
A103 中 川 20 ULL 
A104 渡 边 23 ULL 
A105 西 泽 35 ULL 






































> ( 执行 后 ) 插入 “NI1 1 1” “松田 ” “38” “1980-1 1=1 OO” 










































































empid name age birth 

A101 佐藤 40 ULL 

A102 高 桥 28 ULL 

A103 中 川 20 ULL 

A104 渡 边 23 ULL 

A105 西 泽 35 ULL 

[NE 松田 83 1980-11-10 00:00:00 @ 一 一 一 一 插入 这 条 数据 
操作 方法 





( 执行 下 面 的 命令 。 
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C@) 执行 下 面 的 命 


少 


O 


mysql> INSERT INTO tblC VALUES('N111',' 松 田 ' ,38,'1980-11-10'); 
Query OK, 1 row affected (0.00 sec) 





mysql> SELECT * FROM tbl1C; 













































































吉 字 训 溃 妥 部 总 芋 总 忆 和 这 和 + 
empid name age birth 
本 ee 3 人 + 
Al01 佐藤 40 | NUL 
ARA102 高 桥 28 | NU 
R103 中 20 | NULL 
RA104 渡 边 23 | NU 
RA105 泽 35 | NULL 
N111 松 38 1980-11-10 00:00:00 
i rt a A ee et + 





6 rows in set (0.05 sec) 














我 们 使 用 了 需要 输入 日 期 与 时 间 的 数据 类 型 DATETIME， 所 以 没有 输入 的 时 间 部 分 被 自动 设置 
成 了 “00:00:00”， 即 0 点 0 分 0 秒 。 


6.4 ”修改 列 的 位 置 


像 6.3 节 那 样 执行 “ALTER TABLE ... ADD ...”， 新 建 的 列 会 添加 到 表 的 最 后 面 。 如 果 在 该 命 
令 的 基础 上 加 上 FIRST， 新 建 的 列 就 会 添加 到 最 前 面 。 下 面 的 命令 用 于 将 DATETIME 类 型 的 列 
birth 添加 到 表 tb1D 的 最 前 面 。 


ALTER TABLE tbl1D ADD birth DATETIME FIRST; 





















































使 用 AFTER 能 够 把 列 添加 到 指定 的 位 置 。 例 如 ， 下 面 的 命令 用 于 将 列 birth 添加 到 表 tblE 的 
列 empid 的 后 面 。 


ALTER TABLE tblE ADD birth DATETIME AFTER empid; 


我 们 在 6.2 节 修 改 列 的 数据 类 型 时 使 用 了 MoDIFY (一 6.2 节 )， 其 实 那 时 也 可 以 修改 列 的 
位 置 。 


























@643 修改 列 的 呈 库 


添加 了 列 birth 之 后 ， 试 着 修改 员工 信息 表 tb1C 的 列 的 顺序 。 具 体 来 说 ， 就 是 把 表 tb1C 的 列 
birth 换 到 最 前 面 的 位 置 。 男 外 ， 在 修改 之 后 ， 试 着 把 表 的 列 结 构 显 示 出 来 。 























区 执行 内 容 
> 执行 前 ( 表 tb1C 的 列 结构 ) 





























列 empid name age birth 
数据 类 型 VARCHAR (10) VARCHAR (100) INT DATETIME @ 一 一 一 移动 该 列 














» 


> 执行 后 ( 把 列 birth 换 到 最 前 面 的 位 置 ) 





























列 birth empid name age 
数据 类 型 DATETIME VARCHAR (10) VARCHAR (100) | INT 
操作 方法 


( 执行 下 面 的 命令 。 
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mysql> ALTER TABLE tblC MODIFY birth DATETIME FIRST; 
Query OK, 0 rows affected (0.23 sec) 
Records: 0 Duplicates: 0 Warnings: 0 


mysql> DESC tblC: 














+------- +-------------- +------ +----- +--------- +------- 十 
Field | Type | Null | Key | Default | Extra | 
+------- +-------------- +------ +----- +--------- +------- 十 
birth | datetime | YES | | NUL | | 
empid | varchar(10) | YES | | NULL | 
name | varchar(100) | YES | | NUL | 
age | int (11) | YES | | NULL | | 
+------- +-------------- +------ +----- +--------- +------- 十 
4 rows in set (0.01 sec) 














6.5 ”修改 列 名 和 数据 类 型 


我 们 在 6.3 节 给 表 tb1C 添加 了 新 的 记录 。 其 中 ， 列 birth 保存 的 应 该 是 出 生日 期 ,但 是 却 添加 
了 了 时间“00:00:00”"。 所 以 最 好 还 是 不 使 用 DATETIME 类 型 ， 而 是 使 用 单纯 的 日 期 类 型 DATE。 
下 面试 着 把 列 birth 的 数据 类 型 修改 为 DATE， 并 把 列 名 修改 为 birthday。 























全 6.5.1 修改 列 的 数据 类 型 或 位 置 的 同时 也 修改 列 名 
如 果 想 在 修改 列 的 数据 类 型 或 位 置 的 同时 也 修改 列 名 ， 就 需要 使 用 “ALTER TABLE 


CHANGE ...” 命 邻 。 


修改 列 的 数据 类 型 或 位 置 的 同时 也 修改 列 名 


: ALTER TABLE 表 名 CHANGE 修改 前 的 列 名 修改 后 的 列 名 修改 后 的 数据 类 型 ; 











试 着 把 表 tb1C 的 列 birth 修改 为 DATE 类 型 ， 同 时 把 列 名 修改 为 birthday， 修 改 后 显示 列 的 
结构 。 





区 执行 内 容 
> 执行 前 ( 表 


tb1C 的 列 结构 ) 




































































列 birth empid name age 
数据 类 型 DATETIME VARCHAR (10) VARCHAR (100) INT 
修改 该 列 、 4 
> 执行 后 ( 将 列 birth 的 数据 类 型 修改 为 DATE， 列 名 修改 为 birthday ) 
列 birthday empid name age 
数据 类 型 DATE VARCHAR (10) VARCHAR (100) INT 
操作 方法 























© 


























mysql> ALTER TABLE tblC CHANGE birth birthday DATE; 
Query OK, (0.19 sec) 
Warnings: 0 


6 rows affected 


Records: 6 Duplicates: 0 


mysql> DESC tblC: 




















+---------- +-------------- +------ +----- +--------- +------- 十 
| Field | Type | Null | Key | Default | Extra | 
+---------- +-------------- +------ +----- +--------- +------- 十 
| birthday | date | YES | | NULL | | 
| empid | varchar(10) | YES | | NULL | 

| name | varchar (100) | YES | | NULL | 

| age | int (11) | YES | | NULL | | 
+---------- +-------------- +------ +----- +--------- +------- 十 


4 rows in set (0.05 sec) 








现在 列 birthday 





SELECT 命令 进行 确认 。 


6.6 ”删除 列 





6.6 删除 列 


前 面 我 们 对 表 tb1C 的 列 birthday 进行 了 各 种 修改 ， 最 后 我 们 来 删除 出 生日 期 相关 的 数据 。 


在 SQL 中 , 不 仅仅 是 列 ， 对 数据 库 和 表 执 行 删 除 操作 时 也 会 使 用 DROP 命令 。 
删除 列 


: ALTER TABLE 表 名 DROP 列 名 ; 











P 仅 能 输入 日 期 了 ， 之 前 的 “00:00:00” 等 时 间 数 据 部 分 被 删除 了 。 请 


77 











ban 





试 着 删除 表 tb1C 的 列 birthday， 并 在 删除 后 显示 列 的 结构 。 

































































操作 方法 






































PP 执行 内 容 

> 执行 前 ( 表 tb1C 的 列 结构 ) 

列 birthday empid name age 
数据 类 型 DATE VARCHAR (10) VARCHAR (100) INT 

删除 该 列 、 4 

> 执行 后 

列 empid name age 

数据 类 型 VARCHAR (10) VARCHAR (100) INT 





执行 结果 





mysql> ALTER TABLE tblC DROP birthday; 
Query OK, 0 rows affected (0.69 sec) 
Records: 0 Duplicates: 0 Warnings: 0 


mysgql> DESC tbilc; 
+------- +-------------- 


+------ +----- +--------- +------- 十 
| Field | Type | Null | Key | Default | Extra | 
+------- +-------------- +------ +----- +--------- +------- 十 
| empid | varchar (10) | YES | | NULL | 

| name | varchar(100) | YES | | NULL | 

| age | int (11) | YES | | NULL | | 
+------- +-------------- +------ +----- +--------- +------- 十 
3 rows in set (0.05 sec) 





























如 上 所 示 ， 在 删除 列 的 情况 下 ， 该 列 保存 的 数据 自然 也 会 被 删除 。 拿 上 面 的 例子 来 说 ， 就 是 删 
除了 出 生日 期 的 数据 。 但 是 请 放心 ， 这 项 操作 不 会 影响 到 其 他 列 。 


故意 输入 超过 指定 数量 的 字符 

当 数 据 类 型 指定 为 VARCHAR (10) 时 ， 如 果 输入 了 10 个 以 上 的 字符 ， 会 发 生 什么 情况 呢 ? 假设 表 tb1F 与 
表 tb1 结构 相同 。 我 们 可 以 在 表 tb1F 的 列 name 中 输入 “ 四 四 ”这 16 个 字符 3 
行 试验 。 
























































INSERT INTO tblF (name) VALUES 











在 MySQL 中， 即使 字符 溢出 也 不 会 发 生 错 误 。 执 行 “SELECT * FROM tblF;” 命 令 ， 就 可 以 一 目 了 然 
地 看 到 只 保存 了 “一 二 三 四 一 二 三 四 一 二 ”这 10 个 字符 ， 其 他 数据 都 被 省 略 掉 了 。 人 也 就 是 说 ， 超 过 指定 字符 数 
的 部 分 被 自动 删除 了 。 我 们 需要 注意 避免 出 现 输入 的 数据 在 不 知 不 觉 中 丢失 的 情况 。 

































































CC 








6.7 设置 主键 


@ 6.7.1 什么 是 唯一 


创建 了 数据 库 之 后 ， 就 需要 想 办 法 能 从 大 量 的 数据 中 只 确定 一 个 符合 条 件 的 记录 。 例 如 让 每 个 
员工 都 有 一 个 独一无二 的 会 员 ID ， 或 者 让 一 个 商品 条 码 仅 对 应 一 种 价格 等 。 
这 种 “只 会 确定 一 个 ”的 独一无二 的 状态 ， 称 为 唯一 (unique )。 
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全 6.7.2 什么 是 主键 
创建 唯一 记录 时 ， 会 给 列 设 置 一 个 用 于 和 其 他 列 进行 区 分 的 特殊 属性 。 
在 这 种 情况 下 需要 用 到 的 就 是 主键 (PRIMARY KEY )。 主 键 是 在 多 条 记录 中 用 于 确定 一 条 记 
录 时 使 用 的 标识 符 。 
为 了 可 以 严密 地 确定 某 条 记录 ， 主 键 需 要 具备 以 下 特征 。 






































2 没有 重复 的 值 
2 不 允许 输入 空 值 ( NULL ) 





如 果 在 创建 表 的 时 候 设 置 主键 ， 就 需要 使 用 下 面 的 命令 。 
在 创建 表 的 时 候 设 置 主键 


S 6.7.3 创建 主键 


例如 输入 下 面 的 命令 ， 就 会 创建 表 t pk， 其 中 作为 主键 的 列 a 为 INT 类 型 ， 列 b 为 


VARCHAR (10) 类 型 。 


CREATE TABLE t pk (a INT PRIMARY KEY ,b VARCHAR(10)); 


我 们 可 以 通过 DESC 查看 这 个 表 的 列 结 格 




















执行 结果 

mysql> DESC t pk; 

+------- 和 +------ +----- 人 +------- 十 
| Field | Type | Null | Key | Default | Extra | 
+------- +------------- +------ +----- +--------- +------- 十 
| a | int (11) | NO | PRI | NULL | | 
| b | varchar(10) | YES | | NUL | 
+------- +------------- +------ +----- +--------- +------- 十 
2 rows in set (0.02 sec) 











项 目 Key 中 显示 的 PRI 表示 主键 PRIMARY KEY。 另 外 在 项 目 NULL 中 ， 列 a 显示 为 NO， 这 
表示 不 允许 输入 NULL， 即 不 允许 输入 空 值 。 


6.7.4 ”确认 主键 


例如 ， 我 们 可 以 通过 下 面 的 命令 向 表 t_pk 中 插入 值 为 “1,' 啊 '” 的 记录 。 





INSERT INTO t pk VALUES (1，! 啊 ') ; 


试 着 确认 表 t_pk 的 内 容 。 





mysql> INSERT INTO t pk VALUES (1,' 啊 '); 
Query OK, 1 row affected (0.00 sec) 


mysql> SELECT * FROM t pk; 





| + 
lalp | 
+---+------ + 
11| 啊 | 
+---+------ + 





1 row in set (0.00 sec) 











因为 列 a 中 不 允许 输入 重复 的 值 “1” 和 空 值 “NULL”*， 所 以 在 没有 满足 这 些 条 件 的 情况 下 会 
发 生 下 面 的 错误 。 





mysql> INSERT INTO t pk (a) VALUES (1); 

ERROR 1062 (23000): Duplicate entry '1' for key 'PRIMARY' 
mysql> INSERT INTO t pk (a) VALUES (NULL); 

ERROR 1048 (23000): Column 'a' cannot be null 











当然 ,“1” 和 “NULL” 以 外 的 值 是 可 以 正常 输入 的 。 下 面向 列 a 中 输入 值 “2”。 





mysql> INSERT INTO t pk (a) VALUES (2); 
Query OK, 1 row affected (0.00 sec) 











在 设置 为 主键 的 列 中 ， 不 能 使 用 INSERT 或 UPDATE 命令 输入 已 经 存在 的 值 ( 见 图 6-1 )。 正 
因 如 此 ， 主 键 才能 唯一 标识 一 条 记录 。 

















6.7 设置 主键 











主键 可 以 唯一 标识 一 条 记录 | 



























































不 允许 输入 
NULL 
图 6-1 主键 


6.7.5 “设置 唯一 键 





此 外 ， 我 们 还 可 以 设置 具有 “不 允许 重复 ”这 一 限制 属性 的 唯一 键 (unique key )。 














下 面 创建 表 t_uniq， 该 表 包 括 INT 类 型 的 唯一 键 列 a 和 VARCHAR (10) 类 型 的 列 b。 





CREATE TABLE t uniq (a INT UNIQUE ，b VARCHAR(10)); 


唯一 键 虽然 不 允许 列 中 有 重复 值 ， 但 允许 输入 NULL。 
EE 





mysql> CREATE TABLE t uniq (a INT UNIQUE ,b VARCHAR(10)); 
Query OK, 0 rows affected (0.07 sec) 


mysql> DESC t uniq; 














+------- +------------- +------ +----- +--------- +------- 十 
| Field | Type | Null | Key | Default | Extra | 
+------- +------------- +------ +----- +--------- +------- 十 
| a | int (11) | YES | UNI | NULL | | 
| b | varchar(10) | YES | | NULL | 

+------- +------------- +------ +----- +--------- +------- 十 


2 rows in set (0.03 sec) 
mysql> INSERT INTO t uniq (a) VALUES (NULL); 
Query OK, 1 row affected (0.01 sec) 











通过 上 面 的 执行 结果 ， 我 们 可 以 看 到 项 目 NULL 显示 为 YES， 即 允许 在 列 中 输入 空 值 但 不 允许 





重复 。 





Tr 


6.8 使 列 具 有 自动 连续 编号 功能 


下 面 我 们 试 着 让 列 能 够 从 1 开始 自动 输入 逐次 加 1 的 连续 数字 。 
对 于 名 单 或 者 列表 的 序号 等 ,我们 每 次 都 要 输入 数字 作为 列 的 数据 ， 这 不 仅 麻 烦 ， 还 容易 出 
错 。 所 以 ， 如 果 能 够 自动 输入 1、2、3、4 这 样 的 连续 序号 ， 就 会 方便 许多 。 




















C 6.8.1 具有 自动 连续 编号 功能 的 列 的 定义 


要 使 列 具 有 自动 连续 编号 功能 ， 就 得 在 定义 列 的 时 候 进行 以 下 3 项 设置 。 




















P 数据 类 型 为 INT 等 整数 类 型 
具有 自动 连续 编号 功能 的 列 必须 为 INT、TINYINT 和 SMALLINT 等 整数 类 型 。 既 然 是 连续 编 
， 自 然 为 整数 。 





山 


PP 加 上 AUTO_INCREMENT 


要 给 数据 类 型 INT 加 上 关键 字 AUTO_INCREMENT。 它 用 于 声明 连续 编号 


P 设置 PRIMARY KEY， 使 列 具 有 唯一 性 


具有 自动 连续 编号 功能 的 列 需 要 具有 唯一 性 ， 我 们 可 以 设置 PRIMARY KEY 使 其 变 成 主键 。 





























在 使 用 CREATE TABLE 命令 创建 表 的 时 候 ， 我 们 使 用 了 “a INT,b VARCHAR...” 这 样 的 语 
句 ，AUTO INCREMENT 和 PRIMARY KEY 等 需要 记述 到 数据 类 型 INT 的 后 面 。 

另外 ,设置 为 AUTO_INCREMENT 的 列 自然 是 不 允许 重复 的 ( 唯一 的 状态 )。 这 样 的 列 非常 适 
合作 为 主键 使 用 。 




















C 6.8.2 ”创建 具有 自动 连续 编号 功能 的 列 


我 们 来 试 着 创建 一 个 具有 自动 连续 编号 功能 的 表 t_series 用 于 练习 。 
创建 表 t_series， 该 表 包 括 具有 自动 连续 编号 功能 的 列 a 和 数据 类 型 为 VARCHAR (10) 的 列 b。 
创建 完成 后 ， 显 示 表 的 列 结构 。 
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PP 执行 内 容 
> (执行 后 ) 创建 表 t_series 














列 a b 
INT 

数据 类 型 AUTO INCREMENT VARCHAR (10) 
PRIMARY KEY 














操作 方法 






































DESC t series; 





mysql> CREATE TABLE t series (a INT AUTO INCREMENT PRIMARY KEY,b VARCHAR(10)); 
Query OK, 0 rows affected (0.08 sec) 





mysql> DESC t series; 











+------- +------------- +------ +----- +--------- +---------------- + 
| Field | Type | Null | Key | Default | Extra 

+------- +------------- +------ +----- +--------- +---------------- + 
| a | nt (LLY | NO | PRI | NUL | auto increment | 
| b varchar(10) | YES | | NULL | 

+------- +------------- +------ +----- +--------- +---------------- 十 


2 rows in set (0.02 sec) 


























在 通过 DESC 命令 显示 的 内 容 中 ,项 目 Key 显示 为 PRI， 它 表示 主键 PRIMARY KEY。 男 外 ， 
项 目 Extra 显示 为 auto_increment， 它 表示 设置 了 自动 连续 编号 功能 。 

















其 他 RDBMS 中 自动 连续 编号 功能 的 设置 

动 连续 编号 功能 的 设置 方法 根据 RDBMS 的 不 同 而 不 同 。MySQL 中 是 通过 定义 列 的 AUTO_INCREMENT 

属性 来 实现 的 ， 而 在 Oracle 中 就 不 能 通过 列 的 定义 来 实现 自动 连续 编号 功能 。Oracle 中 可 以 使 用 CREATE 

SEQUENCE 等 命令 ， 通 过 创建 能 够 自动 生成 等 间隔 的 数值 的 序列 ( SEOUENCE ) 来 实现 自动 连续 编号 功能 。 
另外 ，PostgreSOL 中 有 可 以 生成 连续 编号 的 数据 类 型 SERIAL。 但 是 在 使 用 SERIAL 的 情况 下 ， 实 际 上 是 

通过 序列 功能 来 生成 连续 编号 的 。 
















































































































































































6.9 使 用 自动 连续 编号 功能 插入 记录 


下 面试 着 向 表 t_series 中 插入 记录 ， 并 确认 是 否 输入 了 连续 编号 。 因 为 列 a 中 会 自动 输入 连续 
的 编号 ， 所 以 这 次 只 要 在 列 b 中 输入 数据 即 5 



































o 

















要 想 在 设置 了 自动 连续 编号 功能 的 列 中 自动 输入 连续 编号 ， 就 需要 采用 输入 0， 或 者 输入 空 值 
(输入 NULL ) 等 方法 。 


向 表 t_series 中 分 别 插 和 人 值 为 “ 子 ”“ 恬 ”“ 袖 ”的 3 条 记录 。 





P 执行 内 容 


> 执行 前 ( 表 t_series ) 





a b 






























































> 执行 后 

a b 

1 省 

兄 8 一 一 插入 这 3 条 记录 
@ 实 

操作 方法 




















: INSERT INTO t series 
: INSERT INTO t series 
: INSERT INTO t series 


(b) VALUES(' 子 '); 
(b) VALUES(' 丑 '); 
(b) VALUES (' 寅 ') ; 


昌 然 写 的 是 “ 子 ”“ 刁 ”“ 袖 ”这 3 个 值 ， 但 其 实 输 入 任何 值 都 可 以 。 不 要 仅 限 于 这 3 条 记录 ， 
我 们 要 插入 更 多 的 记录 ， 即 使 输入 相同 的 值 也 没有 关系 。 在 需要 反复 执行 相同 命令 的 情况 下 ， 可 以 
先 按 人 键 ， 再 按 Enter 键 ， 反复 执 行 该 操作 (一 3.4.3 节 )。 
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S 6.9.1 确认 插入 的 记录 


适当 搬入 一 些 记 录 后 ,我 们 来 确认 一 下 都 有 哪些 数据 输入 了 进去 。 


SELECT * FROM t series; 




















mysql> SELECT * FROM t series; 
+---+------ 十 

| a | bp | 

+---+------ 十 

| 

1 全 相 于 | 

|13| 寅 | 

二 -一 -十 -一 -一 一 十 

5 rows in set (0.01 sec) 











可 以 看 到 列 a 中 自动 输入 了 从 1 开始 逐次 加 1 的 数据 。 


6.10 ”设置 连续 编号 的 初始 值 


拥有 自动 连续 编号 功能 的 列 还 可 以 设置 任意 的 值 。 

例如 执行 “INSERT INTO t series VALUES (100，' 卯 :' ) ;”， 列 a 中 就 会 输入 值 
“100"”， 然 后 从 “101” 开 始 分 配 连 续 的 编号 ， 即 从 已 经 输入 的 最 大 值 +1 开始 分 配 值 。 由 于 设置 了 
PRIMARY KEY 属性 ， 所 以 不 能 输入 重复 的 值 。 





























如 果 把 这 个 表 中 的 所 有 记录 都 删除 ， 然 后 重新 输入 记录 ， 编 号 会 变 成 什么 样 呢 ? 比如 执行 
DELETE FROM t_series 命令 把 所 有 的 列 都 删除 (一 7.7 节 )， 然 后 执行 “INSERT INTO 七 _ 
series (b) VALUES('xxXx');”。 


可 以 看 到 编号 不 会 重新 从 1 开始 分 配 ， 而 是 从 既 有 最 大 值 +1 的 值 开始 分 配 。 









































执行 结果 

mysql> SELECT * FROM t series; 
+---+------ + 

lalp | 

+---+------ + 

111 子 | 

| 2 | 

131| 寅 | 

sd + 


3 rows in set (0.00 sec) 


mysql> DELETE FROM t series; 
Query OK, 3 rows affected (0.01 sec) 


mysql> INSERT INTO t series (b) VALUES('XX'); 
Query OK, 1 row affected (0.01 sec) 


mysql> SELECT * FROM t series; 





Fs 十 
a | b | 
到 十 
4 | XX | 
i 下 


1 row in set (0.00 sec) 























如 果 想 把 所 有 的 记录 都 删除 掉 ， 并 且 证 编号 从 1 开始 连续 输入 (初始 化 )， 就 需要 按照 下 国 
方式 对 AUTO_INCREMENT 的 值 进行 初始 化 。 


的 























初始 化 AUTO_INCREMENT 的 值 


当 表 中 存在 数据 时 ， 如 果 设 置 的 编号 值 比 已 经 存在 的 值 大 ， 也 可 以 通过 上 面 的 语句 重新 设置 编 
号 的 初始 值 ©。 

另外 ， 正 如 6.9 节 介 绍 的 那样 ， 即 使 向 拥有 自动 连续 编号 功能 的 列 中 输入 0，0 也 不 会 输入 进 
去 。 例 如 执行 “INSERT INTO t series VALUES(0,' 展 ');” 后 ,编号 只 会 根据 规则 连续 输 
入 进去 。 因 此 ， 如 果 总 是 向 拥有 自动 连续 编号 功能 的 列 中 输入 0， 错 误 发 生 的 概率 就 会 降低 。 



























































QD 假设 表 t_series 的 列 a 中 已 经 存在 的 最 大 值 为 4， 这 时 设置 AUTO INCREMENT=10， 下 一 次 编号 的 初始 值 就 是 10。 
(10 比 已 经 存在 的 最 大 值 4 大 ， 可 以 重新 设置 编号 的 初始 值 。) 如 果 设 置 AUTO INCREMENT=1， 下 一 次 编号 的 初 
始 值 就 是 5。( 1 比 已 经 存在 的 最 大 值 4 小 ， 不 能 重新 设置 编号 的 初始 值 。 ) 译 者 注 
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6.11 设置 列 的 默认 值 


本 节 将 会 介绍 给 列 设置 默认 值 (default ) 的 方法 。 在 设置 列 的 默认 值 时 ， 需 要 给 列 加 上 DEFAULT 
关键 字 。 
本 于 二 设置 列 的 默认 值 


: CREATE TABLE 表 名 ( 列 名 数据 类 型 DEFAULT 默认 值 …) ; 

试 着 给 和 员工 信息 表 tbl 具有 相同 结构 的 表 tb1G 设置 列 的 默认 值 。 添 加 “如 果 不 在 姓名 列 
(name ) 中 输入 任何 值 ， 就 会 自动 输入 “未 输入 姓名 ”的 功能 。 

修改 定义 后 ， 试 着 显示 列 的 定义 。 

表 tb1G 的 列 结构 如 表 6-1 所 示 。 





























表 6-1 (执行 前 ) 表 tb1G 的 列 结构 


列 名 数据 类 型 














empid VARCHAR (10) 
name VARCHAR (10) 
age INT 








P 执行 内 容 


PP (执行 前 ) 表 tb1G 的 列 结构 


列 empid name age 























数据 类 型 VARCHAR (10) VARCHAR (10) INT 





PP (执行 后 ) 将 列 name 的 默认 值 设 置 为 “未 输入 姓名 ” 








列 empid name age 
VARCHAR (10) 

据 类 型 VARCHAR (10 INT 
数据 关 10) 全 























区 设置 该 列 











操作 方法 


中 执行 下 面 的 命令 。 


ALTER TABLE tb1G MODIFY name VARCHAR(10) DEFAULT :未 输入 姓名 5 





@ 执行 下 面 的 命令 。 


DESC tb1G; 








mysql> ALTER TABLE tblG MODIFY name VARCHAR(10) DEFAULT ' 未 输入 姓名 '; 
Query OK, 5 rows affected (0.16 sec) 
Records: 5 Duplicates: 0 Warnings: 0 


mysgql> DESC tblG: 


+----- +------------ +------- 十 
| Field | Type | Null | Key | Default | Extra | 
+------- +------------- +------ +----- +------------ +------- + 
| empid | varchar(10) | YES | | NULL | 

| name | varchar(10) | YES | | 未 输入 姓名 | | 
| age | int (11) | YES | | NULL | | 
+------- +------------- +------ +----- +------------ +------- 十 


3 rows in set (0.05 sec) 











C 6.11.2 输入 数据 


按照 计划 修改 好 列 结构 后 ， 试 着 在 列 empid 和 列 age 中 输入 值 ， 不 在 列 name 中 输入 任何 内 容 ， 
看 看 会 出 现 什么 样 的 效果 。 
试 着 输入 员工 ID (empid ) 为 “N999”、 年 龄 (age ) 为 “38” 的 数据 。 


INSERT INTO tblG (empid,age) VALUES ('N999',38); 








mysql> INSERT INTO tblG (empid,age) VALUES ('N999',38); 
Query OK, 1 row affected (0.00 sec) 
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mysql> SELECT * FROM tb1G; 
二 i 下 呈 + 

empid name age 
et i 此 二 
A1l01 佐 芯 40 
A102 高 村 28 
A103 中 川 20 
A1l04 渡 边 323 
RA105 西 泽 35 
N999 未 输入 姓名 38 
Es 二 ee 十 
6 rows in set (0.00 sec) 








列 name 中 输入 了 上 默认 值 “ 未 输入 姓名 ”。 这 是 在 什么 也 不 输入 的 情况 下 设置 的 默认 值 ， 当 然 我 











们 也 可 以 向 列 name 
动 地 保存 进去 。 





数据 库 的 实体 是 什么 

MySQL 创建 的 数据 库 到 底 保存 在 什么 地 方 呢 ? 
MAMP, C:\MAMP\db\mysdql 的 文件 夹 内 就 会 
6-2 )。 








如 果 使 用 本 书 的 方法 安装 了 
件 夹 ， 数 据 库 的 实体 就 保存 在 里 

















( 见 图 
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自动 创建 一 个 和 数据 库 同名 的 文 


1 输入 其 他 任意 的 数据 。 比 如 向 列 name 中 输入 “山田 ”， “山田 ”就 会 原封 不 


6-2 ”CNMAMP\db\mysql 的 内 容 


举例 来 说 ， 当 数据 库 db1 中 存储 了 表 tb 和 tb1 时 , CMAMP\dbmysqlN\db1 的 文件 夹 内 会 有 db.opt、 tb.frm、 
tb.ibd、tb1.frm 和 tb1.ibd 这 5 个 文件 ( 见 图 6-3 )。 














90 | 第 6 章 修改 表 




















= 二 二 
二 ~ 全 部 选择 
才 总 目 加 移动 到 ”| X 开除 > MM rs 
ha 9 ] 口 全 部 
0 国生 到 - 吨 于 名 者 二 FE 
看 
药 贴 板 组 织 新 建 打开 选择 
€ v 个 « db » mysql » dbl1 Y 局 搜索 “db1" 也 
》 图 图 片 名 
》 月 搜索 
2 db.opt tbfrm tbjibd tbl.frm tblibd 
> 
>》 国 桌面 a 
5 个 项 目 用 贺 


图 6-3 文件 夹 db1 内 保存 的 文件 示例 
各 个 文件 的 概况 如 表 6-2 所 示 。 


表 6-2 数据 库 文件 夹 内 的 文件 


文件 名 说 明 














db.opt 记述 默认 字符 编码 等 选项 的 文本 文件 
表 名 .frm 保存 表 的 元 数据 ( 表 定 义 等 ) 的 文件 
表 名 .ibd 保存 在 表 中 的 数据 的 实体 


























但 是 ， 通 过 实际 操作 我 们 可 以 知道 ， 仅 仅 复制 这 些 文件 并 不 能 形成 有 效 的 备份 。 

如 果 自己 在 data 文件 夹 内 创建 一 个 文件 夹 会 出 现 什 么 样 的 结果 呢 ? 我 们 不 妨 试 一 下 。 

比如 在 CNMAMPNdbmysql 的 文件 夹 内 创建 文件 夹 manual， 然 后 在 MySQL 监视 器 上 执行 SHOW 
DATABASES， 这 个 文件 夹 就 会 作为 数据 库 显示 出 来 。 


mysql> SHOW DATABASES; 



























































information schema | 














| 

| wars | 

| manual @ 一 一 一 一 一 -一 一 一 一 一 manual 变 为 了 数据 库 | 
| mysql | 

| performance schema | 

| test | 

+-------------------- 十 


6 rows in set (0.00 sec) 














另外 ， 在 通过 这 种 方式 强制 创建 的 数据 库 中 ， 我 们 也 可 以 照常 创建 表 。 大 家 不 妨 试 一 试 。 











6.12 ”创建 索引 | 91 


6.12 ”创建 索引 


S 6.12.1 什么 是 索引 


当 查 找 表 中 的 数据 时 ， 如 果 数 据 量 过 于 庞大 ， 查 找 操作 就 会 花费 很 多 时 间 。 在 这 种 情况 下 ， 最 
好 在 表 上 创建 索引 ( index )。 
如 果 事 先 在 表 上 创建 了 索引 ， 查 找 时 就 不 用 对 全 表 进 行 扫 描 ， 而 是 利用 索引 进行 扫描 。 这 就 可 
能 会 缩短 查找 时 间 。 

如 果 像 本 书 这 样 只 对 10 行 左 右 的 表 进 行 操作 ， 那 么 有 没有 索引 都 不 会 产生 任何 影响 。 但 是 对 
于 在 企业 中 使 用 的 大 型 表 来 说 就 不 同 了 ， 索 引 的 设置 会 极 大 地 影响 处 理 速度 。 

另外 ， 在 设置 了 主键 (一 6.7 节 ) 的 情况 下 ， 索 引 会 自动 创建 。 

















= 





S 6.12.2 ”创建 索引 


索引 可 以 通过 下 面 的 命令 创建 。 
创建 索引 














在 表 tb1G 的 列 empid 上 创建 名 为 my_ind 的 索引 。 


CREATE INDEX my ind ON tblG (empid); 


mysql> CREATE INDEX my ind ON tblG (empid); 
Query OK, 0 rows affected (0.13 sec) 





Records: 0 Duplicates: 0 Warnings: 0 











创建 的 索引 可 以 通过 下 面 的 命令 显示 出 来 。 


确认 索引 

















下 面 的 执行 结果 中 显示 了 表 tb1G 上 创建 的 索引 。 




















执行 结果 

mysql> SHOW INDEX FROM tblG: 

+------- +------------ +---------- +-------------- +------------- +----------- + 
------- 十 一 一 一 一 一 一 + 
| Table | Non unique | Key name | Seq in index | Column name | Collation | 
Cardi 

nality | Sub part | Packed | Null | Index type | Comment | Index comment | 
+------- +------------ +---------- +-------------- +------------- +----------- 十 
------- +----------+--------+------+------------+---------+---------------+ 
| tblg | 1 | my_ind 1 | empid A 

6 | NULL | NULL | YES | BTREE | | | 
+------- +------------ +---------- +-------------- +------------- +----------- + 
------- +----------+--------+------+------------+---------+---------------+ 


1 row in set (0.00 sec) 








执行 结果 看 起 来 有 些 乱 。 我 们 可 以 把 命令 最 后 的 “;” 换 成 “\G”(G 是 大 写字 母 )， 纵 向 显示 
列 值 ， 这 样 看 起 来 会 更 清晰 一 些 。 





mysql> SHOW INDEX FROM tblG \G 
火炎 火炎 炎炎 大火 火炎 炎炎 大大 炎炎 炎炎 大 大 大 炎炎 炎炎 大 大 了] 。 工 OW 类 炎炎 炎炎 火炎 火炎 炎炎 大火 火炎 炎炎 类 炎炎 类 火炎 类 大 类 类 
Table: tblg 

Non unique: 1 

Key name: my_ind 

Seq in index: 1 

Column name: empid 

Collation: A 

Cardinality: 6 

Sub part: NULL 

Packed: NULL 

Null: YES 

Index type: BTREE 

Comment : 

Index comment: 

1 row in set (0.00 sec) 











创建 好 的 索引 可 以 通过 下 面 的 命令 删除 。 





























医 玫 荆 玫 出 除 索引 


下 面 是 删除 表 tb1G 上 创建 的 索引 my_ina 的 示例 。 


mysql> DROP INDEX my_ind ON tblG: 
Query OK, 0 rows affected (0.08 sec) 





Records: 0 Duplicates: 0 Warnings: 0 











执行 SHOW INDEX 命令， 我 们 能 够 得 知 索引 已 经 被 删除 了 。 


mysql> SHOW INDEX FROM tblG \G 
Empty set (0.00 sec) 














索引 和 处 理 速 度 的 关系 

实际 上 ， 创 建 了 索引 并 不 代表 一 定 会 缩短 查找 时 间 。 因 为 根据 查找 条 件 的 不 同 ， 有 时 候 不 需要 用 到 索引 ， 而 
且 在 某 些 情况 下 ， 使 用 索引 反而 会 花费 更 多 的 时 间 。 
例如 ， 人 们 都 说 在 相同 值 较 多 ( 重复 值 较 多 ) 的 情况 下 最 好 不 要 创建 索引 。 我 们 举 一 个 极端 的 例子 ， 当 某 列 
中 只 有 “YES” 和 “NO” 这 两 个 值 时 ， 即 使 在 该 列 上 创建 索引 也 不 会 提高 处 理 速 度 。 

另外 ， 当 对 创建 了 索引 的 表 进行 更 新 时 ， 也 需要 对 已 经 存在 的 索引 信息 进行 维护 。 所 以 ， 在 使 用 索引 的 情况 
下 ， 检 索 速 度 可 能 会 变 快 ， 但 与 此 同时 ， 更 新 速度 也 很 可 能 会 变 慢 。 
在 使 用 索引 的 情况 下 ， 即 使 索引 在 创建 过 程 中 出 现 了 错误 ， 查 找 结果 也 不 会 受到 任何 影响 。 创 建 索 引 只 会 影 
响 数据 库 整 体 的 处 理 速 度 。 

索引 的 创建 是 影响 整个 数据 库 处 理 效 率 的 重要 问题 。 我 们 把 这 种 提高 处 理 效 率 的 对 策 称 为 调 优 ( tuning )。 
如 何 调 优 就 要 看 数据 库 工程 师 的 技能 了 。 


本 章 介绍 了 以 下 内 容 。 























































































































































































































@ 如 何 修改 表 的 列 结构 

@ 什 么 是 主键 

@ 如 何 设置 拥有 自动 连续 编号 功能 的 列 
@ 如 何 设置 列 的 默认 值 


@ 索 引 的 作用 和 创建 方法 














在 数据 库 中 ,“ 唯 一 值 ”很 重要 。 我 们 要 学 会 使 用 PRIMARY KEY、UNIQUE 、 自 动 连续 编号 功 








能 和 索引 。 


> 自我 检查 











下 面 检查 一 下 本 章 学 习 的 内 容 是 否 全 部 理解 并 掌握 了 。 











口 能 够 使 用 “ALTER TABLE ...” 命 令 对 列 结构 进行 修改 、 添 加 和 删除 
口 能 够 给 列 设置 自动 连续 编号 功能 。 能 够 输入 连续 的 编号 

口 能 够 使 用 “AUTO INCREMENT=...” 设 置 连续 编号 的 初始 值 

口 能 够 使 用 DEFAULT 设置 列 的 默认 值 

口 能 够 使 用 “CREATE INDEX ...” 创 建 索 引 

口 能 够 显示 索引 的 内 容 并 删除 索引 





i 练习 题 








问题 1 
创建 列 结构 如 表 6-3 所 示 的 表 t_initial_serial， 然 后 把 输入 时 的 时 间 保 存 到 列 c_time 中 并 插入 记录 。 


























表 6-3 表 t_initial_serial 














具体 内 容 连续 编号 输入 时 间 
列 名 id c_time 
INT 类 型 
ee AUTO INCREMENT 本 
数据 类 型 二 DATETIME 类 型 
PRIMARY KEY 
初始 值 为 100 

















HINT] 可 以 使 用 NOw 函数 获取 当前 时 间 。 把 NOW () 作为 数据 输入 ， 就 能 够 输入 当前 时 间 了 。 函 
ls 数 相关 的 内 容 会 在 第 8 章 进行 介绍 。 详 细 内 容 可 参考 8.2.6 节 。 












































执行 下 面 的 命令 。 
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CREATE TABLE 七 initial _ serial (id INT AUTO INCREMENT PRIMARY KEY,c time 
DATETIME) AUTO INCREMENT=100; 


INSERT INTO 七 i 本 e) VALUES (NOW()); 





mysql> CREATE TABLE 七 Initial serial (id INT AUTO INCREMENT PRIMARY KEY,c time 
DATETIME) AUTO INCREMENT=100; 


Query OK, 0 rows affected (0.09 sec) 





mysql> INSERT INTO 七 initial serial (c time) VALUES (NOW()); 
Query OK, 1 row affected (0.05 sec) 


mysql> SELECT * FROM t initial serial; 


二 二 本 
| ida | c time | 
人 相 十 
| 100 | 2018-05-15 19:59:23 | 
+----- 和 + 


1 row in set (0.00 sec) 











第 7 章 复制、 删除 表 和 记录 


本 章 将 介绍 如 何 复制 表 的 列 结构 和 记录 ， 以 及 如 何 完全 删除 
数据 库 、 表 和 记录 。 











7.1 复制 表 的 列 结构 和 记录 


下 面 将 介绍 表 和 记录 的 删除 方法 以 及 各 种 提取 方法 。 因 为 需要 用 到 很 多 表 ， 每 次 输入 记录 又 很 
麻烦 ， 所 以 为 了 能 重复 利用 之 前 创建 过 的 表 ， 我 们 要 掌握 表 的 各 种 复制 方法 。 
接 下 来 会 介绍 以 下 3 种 复制 方法 。 



































2 复制 表 的 列 结构 和 记录 
2 仅 复 制 表 的 列 结构 
2 仅 复 制 记录 


7.2 ”将 表 的 列 结 构 和 记录 整个 复制 过 来 

















先 介绍 第 1 种 方法 ， 也 就 是 使 用 SELECT 的 结果 复制 列 结 构 和 记录 ， 然 后 创建 新 的 表 。 该 方法 
可 以 将 包括 记录 在 内 的 整个 表 复 制 过 来 ， 非 常 方 便 。 

但 是 ， 这 种 方法 不 能 复制 AUTO_INCREMENT 等 属性 。AUTO_INCREMENT 等 属性 需要 在 复制 
后 再 次 进行 设置 。 
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7.2.1 复制 表 的 列 结构 和 记录 


2 
相关 语法 非常 容易 理解 ， 也 就 是 使 用 SELECT 的 记录 执行 CREATE TABLE。 


复制 表 的 列 结构 和 记录 来 创建 表 





记录 。 


使 用 “SELECT * FROM ...” 的 结果 执行 CREATE TABLE。 这 样 在 创建 新 表 的 同时 也 复制 了 


在 4.9 方 ， 我 们 创建 了 很 多 表 tb1 的 副本 ， 当 时 使 用 的 就 是 这 个 语法 。 下 面试 着 通过 复制 表 tb1l 


来 创建 表 tbl_bk。 





PP 执行 内 容 


> 执行 前 ( 表 tb1 的 列 ) 
































兮 复 制 这 个 结构 




















一 一 创建 该 列 



































列 empid name age 
数据 类 型 VARCHAR (10) VARCHAR (10) INT 
> 执行 后 ( 通过 复制 创建 的 表 tb1_bk 的 列 ) 
列 empid name age 
数据 类 型 VARCHAR (10) VARCHAR (10) INT 
操作 方法 
执行 下 面 的 命令 。 























试 着 执行 下 面 的 命令 来 确认 一 下 表 是 否 被 正确 地 复制 了 。 

















ey 
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mysql> SELECT * FROM tbl bk; 
ss 下 和 各 训 和 有 各 和 十 
empid name age 
es rs es 十 
A1l01 佐 有 芋 40 
RA102 高 村 28 
A1l03 中 川 20 
A104 渡 边 23 
RA105 泽 35 
于 十 
5 rows in set (0.00 sec) 








记录 被 正常 复制 过 来 了 。 如 果 用 8.3.2 节 介 绍 的 WHERE 或 8.3.1 节 介 绍 的 LIMIT 来 限定 要 提取 
的 记录 ， 还 可 以 只 对 需要 的 记录 进行 复制 。 


另外 ， 这 种 方法 可 能 会 改变 列 的 属性 。 例 如 在 某 些 MySQL 的 版 本 中 ，VARCHAR (100) 可 能 会 


能 会 
变 成 CHAR (100) 。 除 此 之 外 ， 还 可 能 存在 不 复制 元 表 的 索引 等 情况 。 所 以 在 执行 完 复 制 操作 后 ， 


请 用 DESC 确认 表 的 结构 ， 然 后 再 使 用 该 表 。 



































7.3” 仅 复制 表 的 列 结构 


下 面 将 介绍 通过 复制 表 的 列 结构 来 创建 表 的 方法 。 这 种 方法 虽然 不 会 复制 表 中 的 记录 ， 但 是 会 
复制 AUTO_INCREMENT 和 PRIMARY KEY 等 列 的 属性 








7.3.1 仅 复 制 表 的 列 结构 


人 一 一 一 一 一 一 一 一 


在 CREATE TABLE 命令 的 表 名 后 面 加 上 LIKE 指定 复制 的 元 表 。 





E 包 呈 | 通过 复制 表 的 列 结构 来 创建 新 表 




















接 下 来 ,我 们 试 着 创建 一 个 与 员工 信息 表 tbl 具有 相同 列 结构 的 空 表 tb1 _bkc， 并 且 试 着 显示 表 
tbl 和 表 tbl_bkc 的 列 结构 。 
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P 执行 内 容 
> 执行 前 ( 表 tb1 的 列 ) 




















列 empid name age 一 一 一 一 一 复制 列 结 构 
数据 类 型 VARCHAR (10) VARCHAR (10) INT 




















D4 


> 执行 后 ( 通过 复制 列 结构 创建 的 表 tb1_bkc ) 









































列 empid name age 一 一 一 一 创建 该 表 
数据 类 型 VARCHAR (10) VARCHAR (10) INT 
操作 方法 






































DESC tbl; 














+------- +------------- +------ +----- +--------- +------- 十 
| Field | Type | Null | Key | Default | Extra | 
+------- +------------- +------ +----- +--------- +------- + 
| empiqd | varchar(10) | YES | | NULL | 

| name | varchar(10) | YES | | NULL | 

| age | int (11) | YES | | NULL | | 
+------- +------------- +------ +----- +--------- +------- 十 


3 rows in set (0.08 sec) 


mysql> DESC tbl bkc; 








+------- +------------- +------ +----- +--------- +------- + 
| Field | Type | Null | Key | Default | Extra | 
+------- +------------- +------ +----- +--------- +------- + 
| empiqd | varchar(10) | YES | | NULL | 

| name | varchar(10) | YES | | NULL | 

| age | ne:(LL) | YES | | NULL | | 
+------- +------------- +------ +----- +--------- +------- 十 


3 rows in set (0.07 sec) 
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该 方法 也 会 复制 AUTO_INCREMENT 和 PRIMARY KEY 等 列 的 属性 "。 这 是 一 种 不 复制 记录 ,只 
复制 列 结构 的 方法 。 


7.4 复制 其 他 表 的 记录 


下 面 介绍 如 何 向 创建 好 的 表 中 复制 其 他 表 的 记录 ( 数据 )。 





5 7.4.1 复制 其 他 表 的 记录 
我 们 可 以 使 用 下 面 的 方法 复制 具有 相同 列 结构 的 表 的 记录 。 具 体 语法 如 下 所 示 。 


复制 其 他 表 的 记录 


oooooosssssosseoseesssssoosos eossusssooso ososeuos ooo oes aosesso ooo ooososeesoo oo os oseeeoe eosoosssseeos esos esse ose oresossseoseeeessses 


esoseooesosssoseeooesseosoooo eososeo ooo ses eee see 00000 0 0】 0000000 0000 00 0.0. 


在 7.3 节 中 ,我 们 创建 了 一 个 和 表 tbl 具有 相同 结构 的 空 表 tbl_bkc。 现 在 试 着 将 表 tbl 中 的 所 
有 记录 都 复制 到 表 tb1_bkc 中 。 搬 和 记录 后 ， 将 所 有 的 记录 显示 出 来 。 


> 执行 内 容 


> 执行 前 ( 表 tb1_bkc ) 





empid name age 























empid name age 





从 表 tb1 中 复制 
了 所 有 的 记录 




















中 这 一 点 可 以 通过 第 6 章 中 介绍 的 表 t_series 进行 验证 。 一 一 译 者 注 


7.5 选择 某 一 列 进行 复制 | 101 


操作 方法 


4) 执行 下 面 的 命令 。 




















我 们 可 以 确认 一 下 所 有 的 记录 是 否 都 插入 到 了 表 tbl_bkc 中 。 












































mysql> SELECT * FROM tbl bkc; 

+------- 二 -一 -一 一 一 +------ 十 
empid name age 

+------- 二 -一 -一 一 一 +------ 十 
和 RA101 佐 茧 40 
A1l02 高 李 28 
RA103 中 川 20 
Al04 渡 边 23 
A1l05 泽 35 

吾 且 部 台 学 学 学 总 dd et 二 

5 rows in set (0.00 sec) 








7.5 选择 某 一 列 进行 复制 





我 们 可 以 从 元 表 中 选择 某 一 列 的 记录 进行 复制 。 例 如 下 面 的 示例 就 展示 了 如 何 向 表 tbl_bkc 的 
姓名 列 name 中 插入 表 tbl 的 列 empid 的 记录 。 


INSERT INTO tb1 bkc (name) SELECT empid FROM tb1; 


只 有 表 tb1_bkc 的 列 name 中 输入 了 员工 号 ， 其 他 列 中 都 输入 了 NULL。 
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执行 结果 
mysql> SELECT * FROM tbl bkc 
ss 下 和 各 这 入 有 各 和 + 
empid name age 
hs ES RE 十 
A1l01 佐藤 40 
RA102 高 村 28 
RA103 中 川 20 
A104 渡 边 23 
RA105 西 泽 35 
NULL R101 | NU 
NU A102 NU 
NU A103 NU 
NU R104 | NU 
NU RAR105 | NU 
ES Ee 十 
10 rows in set (0.00 sec) 

















在 这 种 情况 下 ， 因 为 列 empid 和 列 name 的 数据 类 型 都 是 VARCHAR (10) ， 所 以 命令 的 执行 没 
有 出 现任 何 问题 。 但 是 ， 如 果 数 据 类 型 不 一 致 ， 复 制 操 作 就 可 能 会 失败 ， 这 一 点 需要 大 家 注意 。 
另外 ,我们 还 可 以 使 用 WHERE (一 8.3.2 节 ) 复制 符合 条 件 的 记录 ,使 用 LIMIT (一 8.3.1 节 ) 
指定 插入 的 记录 数 。 





























7.6 ”删除 表 、 数 据 库 和 记录 


接 下 来 介绍 删除 表 、 数 据 库 和 记录 的 方法 。 记 录 删 除 后 大 多 无 法 复原 ， 所 以 在 执行 DROP 或 
DELETE 命令 的 时 候 一 定 要 慎重 。 








虞 | 





7.7 ”删除 表 


下 面 是 删除 表 的 方法 。 我 们 可 以 使 用 DROP 命令 删除 表 。 
删除 表 





























试 着 删除 表 tp1A， 并 在 删除 后 显示 当前 存在 的 所 有 表 的 表 名 。 





P 执行 内 容 


P> 执行 前 ( 表 tb1A ) 
























































empid name age 
A101 佐 爱 40 
A102 高 桥 28 
A103 中 川 20 
A104 渡 边 23 
A105 泽 35 
操作 方法 























@ 一 一 一 一 圳 除 该 表 






































mysql> SHOW TABLES; 


13 rows in set 





(0.01 sec) 
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表 tblA 消失 了 ， 再 也 不 能 回 到 原来 的 状态 了 。 在 执行 包含 DROP 的 命令 时 一 定 要 谨慎 。 





如 下 所 示 ， 在 DROP TABLE 的 后 面 加 上 IF EXISTS， 就 表示 如 果 表 tb1A 存在 就 将 其 删除 。 




















DROP TABLE IF EXISTS tblA; 


一 般 来 说 ,在 目标 表 不 存在 的 情况 下 执行 DROP 命令 会 发 生 错误 ,但 如 果 加 上 了 IF EXISTS， 
就 能 够 抑制 错误 的 发 生 。 

例如 14.2 节 会 介绍 如 何 使 用 SoURCE 命令 执行 记述 了 多 个 SQL 语句 的 文本 文件 。 这 时 如 果 在 
命令 中 加 上 IF EXISTS， 无 论 表 存 在 与 否 , 命令 都 能 执行 ， 非 常 方便 。 





7.8 删除 数据 库 


删除 数据 库 时 也 需要 使 用 DROP 命令 。 执 行 “DROP DATABASE db0;”， 数 据 库 db0 就 会 被 删除 。 





删除 数据 库 





执行 上 面 的 命令 之 后 ， 数 据 库 就 无 法 恢复 到 原来 的 状态 了 。 包 括 表 在 内 的 所 有 信息 都 将 消失 ， 
这 一 点 需要 特别 注意 。 另 外 ， 现 在 使 用 的 数据 库 dbl 在 本 书后 面 的 内 容 中 也 会 用 到 ， 注 意 不 要 把 它 
删除 了 。 

















7.9 删除 所 有 记录 


下 面 将 介绍 不 删除 表 自 身 ， 只 删除 表 中 记录 的 方法 。 这 里 只 介绍 删除 所 有 记录 的 方法 。 
删除 所 有 记录 





执行 “DELETE FROM 表 名 ;”， 表 里 的 所 有 记录 都 会 被 删除 。 另 外 ， 如 果 想 指定 记录 进行 删 
除 ， 可 以 使 用 WHERE 来 设置 条 件 。 关 于 WHERE， 请 参考 8.3.2 节 的 内 容 。 
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试 着 删除 表 tp1_bk 里 的 所 有 记录 ， 并 且 在 删除 后 确认 记录 是 否 已 经 消失 。 


P 执行 内 容 
PF 执行 前 ( 表 tb1_bk ) 





empid name age 





@ 一 一 一 一 删除 记录 














> 执行 后 4 


empid name age 















































操作 方法 


QQ 执行 下 面 的 命令 。 





DELETE FROM tbl1 bk; 


@ 执行 下 面 的 命令 。 






SELECT * FROM tbl1 bk; 


mysql> DELETE FROM tbl bk; 
Query OK, 5 rows affected (0.01 sec) 





mysql> SELECT * FROM tbl1 bk; 
Empty set (0.00 sec) 
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执行 SELECT 后 会 显示 Empty set， 由 此 可 知 ， 所 有 的 记录 都 被 删除 了 。 


不 使 用 MySQL 监视 器 操作 MySQL 
我 们 可 以 直接 从 命令 提示 符 或 终端 来 操作 MySQL。 





人 @@ 使 用 mysqladmin 命令 创建 和 删除 数据 库 
例如 MySQL 里 提供 了 一 个 名 为 mysqladmin 的 强大 命令 来 创建 数据 库 。 我 们 可 以 执行 mysqlaqdmin 命令 
创建 数据 库 。 这 个 命令 不 是 MySQL 监视 器 的 命令 ， 所 以 请 直接 从 命令 提示 符 或 终端 输入 。 






































转 使 用 mysqladmin 创建 数据 库 























Boosossssessssoooossosseossooosooooeessosoosossssessesoossosoeooseeoossosoeososeeasssos eosoeeesooooseoo esse oesesooooooss 












































对 于 用 户 名 和 密码 ， 请 输入 和 MySQL 监视 器 启动 时 相同 的 内 容 。 下 面 是 在 命令 提示 符 中 使 用 mysqladmin 
命令 创建 数据 库 db2 的 示例 。 


mysqladmin -u root -proot CREATE db2 


这 样 就 创建 好 了 数据 库 db2。 
接 下 来 将 介绍 使 用 mysqladmin 命令 删除 数据 库 db2 的 方法 。 执 行 该 命令 后 会 显示 是 否 删 除数 据 库 的 确认 
信息 ， 请 按 “ YY o 


mysqladmin -u root -proot DROP db2 


全 使 用 mysql 命令 执行 查询 




















































































































































































































虽然 只 要 使 用 mysql 命令 就 能 启动 MySQL 监视 器 ， 但 是 也 可 以 在 不 启动 MySQL 监视 器 的 情况 下 直接 执 
行 SOL 语句 。 在 这 种 情况 下 ， 需 要 加 上 -e 选项 ， 并 且 用 "" 将 后 面 的 命令 括 起 来 。 注 意 不 是 用 单 引 号 ( '… )， 
而 是 用 双 引 号 ( "" ) 把 命令 括 起 来 。 

















国 使 用 mysql 命令 执行 SQL 语句 















































下 面 是 使 mysql 命令 直接 执行 SQL 语句 的 示例 。 


mysql dbl -u root -proot -e "SELECT * FROM tb1" 


在 之 后 的 章节 里 ， 除 了 上 面 介绍 的 内 容 之 外 还 会 出 现 各 种 不 使 用 MySQL 监视 器 的 操作 ， 比 如 使 用 
mysqldump 命令 获取 数据 库 的 dump 文件 (一 14.4.1 节 )， 或 者 使 用 mysql 命令 进行 复原 ( 一 14.4.3 
) 等 




































































多 行 输入 
前 面 介绍 的 SOL 语句 非常 简单 ， 所 以 我 们 把 所 有 内 容 都 输入 在 了 1 行 中 。 比 如 下 面 这 个 语句 (一 4.8.1 节 )。 


SELECT empid, name FROM tbl; 


可 是 ， 命 令 变 长 后 易 读 性 就 会 大 打折 扣 ， 还 会 出 现 1 行 无 法 写 完 的 情况 。 在 初始 化 设置 中 ， 输入“;” 后 命 
令 才 能 执行 ， 因 此 没有 必要 强行 把 命令 写 在 1 行 中 。 不 妨 通过 换行 让 命令 变 得 更 易于 阅读 ， 然 后 在 命令 的 最 后 加 
EE 

另外 ，SQL 语句 中 可 以 输入 tab 或 半角 空格 。 也 就 是 说 ， 可 以 随意 缩 进 到 更 容易 阅读 的 位 置 。 但 是 请 注意 ， 
如 果 在 关键 字 或 记录 的 中 间 进 行 换 行 就 会 出 错 。 比 如 对 于 SELECT， 就 不 能 在 SEL 和 ECT 的 中 间 进 行 换 行 。 

对 上 面 的 SQL 语句 执行 换行 和 缩 进 后 就 变 成 了 下 面 这 样 。 








































































































































































































SELECT 


empid, name 


FROM 
tb1l 








如 果 不 输入 最 后 的 “ ;”， 命 令 就 不 会 执行 。 一 边 输 入 一 边 确认 输入 内 容 可 以 减少 错误 的 发 生 。 

如 果 每 一 行 都 执行 换行 ， 不 仅 阅 读 起 来 更 加 方便 ， 也 更 容易 发 现 错误 。 另 外 ， 当 SQL 语句 像 14.2 节 那 样 写 
在 文本 文件 中 时 ， 我 们 也 可 以 随意 对 其 进行 改写 ， 非 常 方便 。 

本 书 原 本 也 打算 采用 换行 这 种 更 易于 理解 的 方式 进行 书写 ， 但 因 页 数 限制 ， 最 终 只 在 必要 的 时 候 执行 了 换 
行 ， 大 部 分 命令 写 在 了 1 行 中 。 大 家 可 以 适当 地 进行 换行 、 缩 进 ， 使 输入 的 内 容 更 易于 阅读 。 
但 是 严格 来 说 ， 不 管 是 半角 空格 还 是 tab， 输 入 的 其 实 都 是 代码 ， 如 果 执 行 了 缩 进 的 SOL 语句 变 成 文件 ， 文 
件 自然 也 会 变 大 。 不 过 现在 硬盘 容量 和 处 理 能 力 已 经 不 再 是 限制 条 件 了 ， 文 件 变 大 所 造成 的 影响 基本 可 以 忽略 不 
计 。 易 于 理解 的 描述 方式 是 防止 错误 的 有 效 手 段 ， 请 一 定 设计 出 方便 自己 理解 的 写法 。 















































































































































































































































7.10 ”总结 


本 章 介绍 了 以 下 内 容 。 


@ 如 何 通过 复制 其 他 表 的 列 结构 和 记录 来 创建 表 
@ 如 何 通过 复制 其 他 表 的 列 结构 来 创建 表 

@ 如 何 复制 其 他 表 的 记录 

@ 如 何 删除 表 、 数 据 库 和 记录 





我 们 在 第 9 章 会 学 习 如 何 通过 WHERE 等 命令 来 设置 条 件 进 行 复制 和 删除 。 我 们 先 记 住 复 制 和 
删除 所 有 记录 的 方法 。 至 此 ， 我 们 已 经 能 够 轻松 创建 用 于 本 书 学 习 的 素材 了 。 
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> 自我 检查 
下 面 检查 一 下 本 童 学 习 的 内 容 是 否 全 部 理解 并 掌握 了 。 

















口 能 够 使 用 “CREATE TABLE ... SELECT ...” 复 制 列 结构 和 记录 来 创建 表 
口 能 够 使 用 “CREATE TABLE ... LIKE ...” 复 制 列 结构 来 创建 表 
口 能 够 使 用 “INSERT INTO ... SELECT ... FROM ...” 复 制 表 里 的 记录 
口 能 够 使 用 DROP 删除 数据 库 、 表 和 记录 





i 练习 题 





| 问题 1 
表 t_name 的 列 结构 如 表 7-1 所 示 。 请 将 表 tb1 的 列 name 的 记录 插入 进去 。 























表 7-1 表 t_name 























列 的 数据 类 型 VARCHAR (10) 
松 尾 
市 川 
列 的 内 容 
伊 世 
内田 






































执行 下 面 的 命令 。 


INSERT INTO t name(a) SELECT name FROM tb1; 
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5 rows in set (0.00 sec) 











只 复制 列 name 的 记录 。 


mysql> INSERT INTO t name(a) SELECT name FROM tbl: 
Query OK, 5 rows affected (0.02 sec) 





Records: 5 Duplicates: 0 Warnings: 0 


mysql> SELECT * FROM t name; 


















































10 rows in set (0.00 sec) 
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第 3 部 分 
熟练 使 用 MySOL 


Oo 


使 用 各 种 条 件 进行 提取 
编辑 数据 

EU 使 用 多 个 表 
熟练 使 用 视图 

第 熟练 使 用 存储 过 程 
ED 熟练 使 用 事务 


Bg 使 用 文件 进行 交互 























只 是 简单 地 使 用 INSERT、SELECT 和 DELETE 不 可 能 完全 操作 数据 库 。 实 际 工作 中 通常 
会 使 用 多 个 复杂 关联 在 一 起 的 表 来 处 理 大 量 数据 ， 这 时 不 仅 要 考虑 处 理 效率 ， 还 要 避免 意外 
丢失 数据 的 情况 发 生 。 

MySQL 的 学 习 也 将 迎 来 佳境 。 从 如 何 设置 条 件 进 行 提 取 和 编辑 ， 到 如 何 使 用 具有 关联 关 
系 的 多 个 表 ， 这 一 系列 过 程 都 会 在 接 下 来 的 内 容 中 学 到 。 我 们 还 会 介绍 一 些 在 实践 中 需要 用 
到 的 内 容 ， 如 视图 、 存 储 过 程 、 事 务 和 文件 操作 等 。 
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8.1 


这 种 简单 的 SELECT 命令 来 显示 记录 。 但 对 


























面 的 内 容 中 ， 我 们 使 用 的 都 是 “SELECT * FROM tbl;” 






































实际 使 用 数据 库 时 ， 





















































显示 所 有 












































记录 和 所 有 列 是 不 够 的 ， 必 须 使 








SELECT 是 SOL 的 基础 。 明 确 各 个 数 和 














居 








各 种 手段 快速 找到 











要 以 怎样 的 形式 提 












































口 ,， 





操作 sELECT 是 我 们 要 达到 的 


设计 列 的 显示 内 容 并 执行 SELECT 











标 。 


8.1.1 准备 表 tb 
从 本 音 开 始 ， 我 们 将 使 用 销售 信息 表 tb 来 练习 SELECT 的 用 法 。 表 tb 中 保存 了 各 员工 的 销售 
和 额 和 相关 月 份 的 数据 。 请 使 用 附录 3 事先 创建 好 这 个 表 。 



































> 表 tb 的 内 容 

empid sales month 
A103 101 4 
A102 54 5 
A104 181 4 
A101 184 4 
A103 17 5 
A101 300 5 
A102 205 6 
A104 93 5 
A103 12 6 
A107 87 6 
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表 也 由 员工 号 (empid )、 销 售 额 (sales ) 和 月 份 (month ) 3 个 列 组 成 。 通 过 关联 的 员工 信息 
表 tbl 可 以 知道 , 表 也 中 empid 为 A101 的 员工 是 佐 蔷 。 
下 面 介绍 的 内 容 都 以 启动 了 MySQL 监视 器 并 执行 了 use abl 为 前 提 。 








在 之 前 的 内 容 中 ,我 们 使 用 了 “SELECT * FROM tb;” 这 样 的 命令 来 查找 记录 。 该 命令 表示 
从 (FROM ) tb 中 选择 ( SELECT ) 所 有 的 列 (* ),。“*” 是 通配符 (wildcard )， 可 以 替代 任意 字符 。 

在 指定 了 列 名 的 情况 下 ， 列 会 按照 指定 的 顺序 显示 (一 4.8 节 )。 当 指定 多 个 列 时 ， 可 以 使 用 
“，” 来 分隔 列 名 。 

我 们 让 员工 信息 表 tb 只 包含 销售 额 (sales ) 和 员工 号 (empid ) 的 信息 ， 并 按照 这 个 顺序 显示 
出 来 。 














> 执行 内 容 



































































































































> 执行 前 ( 表 tb ) > 执行 后 
empid sales month sales empid 
A103 101 4 101 A103 
A102 54 5 54 A102 
A104 181 4 81 A104 
A101 184 4 184 A10 
A103 17 5 中 17 A103 
A101 300 5 300 A101 
A102 205 6 205 A102 
A104 93 5 93 A104 
A103 12 6 12 A103 
A107 87 6 87 A107 
这 两 列 调换 顺序 后 显示 出 来 
操作 方法 
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执行 结果 
mysql> SELECT sales,empid FROM tb; 
二 = 于 写字 生字 二 党 二 + 
sales empid 
和 SEE SS 十 
101 A1l03 
54 A1l02 
181 Al04 
184 Al01 
4147 A1l03 
300 Al01 
205 A1l02 
93 Al04 
下 2 A103 
87 Al107 
二 SEE SEE 十 
10 rows in set (0.00 sec) 











同一 个 列 可 以 显示 多 次 。 例 如 执行 命令 “sELECT sales,empid,sales,empid,sales, 
empid FROM tb;”， 列 sales 和 列 empid 就 会 各 显示 3 次 。 














® 8.1.3 ”使 用 别名 | 


即使 显示 了 列 名 empid 和 sales， 一般人 也 不 太 清 楚 它 表示 的 是 什么 意思 。 如 果 给 这 些 列 加 上 
“昵称 ”"， 理 解 起 来 就 容易 多 了 。 

这 个 昵称 就 是 别名 (alias )。 别 名 是 指 一 般 称谓 以 外 的 名 称 。 在 计算 机 的 世界 里 ， 它 表示 为 了 
指 代 真实 事物 ， 由 用 户 自 由 命名 的 名 称 。 

如 果 执 行 “SELECT * FROM tb;”, 第 1 行将 显示 出 真正 的 列 名 empid、sales 和 month。 此 
时 我 们 也 可 以 将 其 蔡 换 为 容易 理解 的 别名 。 

指定 别名 


















































在 指定 多 个 列 的 情况 下 ， 需 要 使 用 “, ”区 分 各 列 名 ， 并 加 上 “ 列 名 As 别名 ”。 如 果 别 名 使 
用 了 特殊 符号 ， 就 需要 使 用 " " 将 别名 括 起 来 。 

下 面试 着 给 列 empid 加 上 别名 “员工 号 "， 给 列 sales 加 上 别名 “销售 额 "*， 然 后 把 表 tb 中 的 所 
有 记录 都 显示 出 来 。 
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P 执行 内 容 
> 执行 前 ( 表 tb ) > 执行 后 
empid sales month 员工 号 销售 额 
A103 101 4 A103 101 
A102 54 5 A102 54 
A104 181 4 A104 181 
A101 184 4 A101 184 
A103 17 5 A103 17 
A101 300 5 A101 300 
A102 205 6 A102 205 
A104 93 5 A104 93 
A103 12 6 A103 12 
A107 87 6 A107 87 
这 两 列 加 上 别名 后 显示 出 来 
操作 方法 
执行 下 面 的 命令 。 
SELECT empid RS 员工 号 ,sales RS 销售 额 FROM tb; 
执行 结果 
mysql> SELECT empid AS 员工 号 ,sales AS 销售 额 FROM tb; 
开本 二 已 吕 忆 党 全 二 二 + 
员工 号 销售 额 
rr 4 + 
A1l03 于 0 于 
A1l02 54 
A1l04 181 
A1l01 184 
A1l03 lg 
Al01 300 
A1l02 205 
A1l04 93 
A1l03 le 
A1l07 87 
和 兴 芝 人 + 
10 rows in set (0.00 sec) 
比 以 前 更 容易 理解 了 吧 。 如 果 把 显示 出 来 的 字符 串 复 制 粘 贴 到 某 个 文档 中 ， 这 些 字符 串 就 可 以 








直接 作为 基础 数据 使 用 了 。 
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8.2 ”计算 列 值 或 处 理 字 符 串 之 后 显示 列 


S 8.2.1 使 用 列 值 进行 计算 并 显示 
我 们 可 以 使 用 列 中 的 数据 自由 地 进行 乘法 和 除法 等 四 则 运算 。 但 在 计算 机 的 世界 中 ， 不 能 直接 
使 用 符号 x 或 * ， 而 要 使 用 表 8-1 中 列 出 的 算术 运算 符 。 


























表 8-1 算术 运算 符 





























运算 符 使 用 示例 含义 

十 a + b a 加 上 b 

- a-b a 减 去 b 

* a*b a 乘 以 b 

/ a/b a 除 以 b 

DIV a DIV b a 除 以 b ( 结果 取 整 ) 
$$、MOD asgspb a 除 以 b 取 余 








假设 销售 信息 表 tb 中 的 列 sales 的 值 以 “万 元 ”为 单位 。 那 么 ,“A101” 估 了 蔷 5 月 份 的 销售 额 
为 300 万 元 ( 表 了 b 的 第 6 行 数据 )。 
试 着 给 表 tb 的 列 sales 的 值 乘 以 10000 并 给 该 列 添加 别名 “销售 额 ”， 然 后 显示 所 有 记录 。 








> 执行 内 容 





































































































> 执行 前 ( 表 tb ) > 执行 后 

empid sales month 销售 额 @ 一 一 一 给 列 名 加 上 别名 
A103 101 4 010000 

A102 54 5 540000 

A104 181 4 810000 

A101 184 4 840000 

A103 17 5 70000 将 信条 以 10000 
A101 300 5 3000000 后 显示 出 来 
A102 205 6 2050000 

A104 93 5 930000 

A103 12 6 120000 

A107 87 6 870000 
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操作 方法 























1010000 
540000 
1810000 
1840000 
170000 
3000000 
2050000 
930000 
120000 
870000 








10 rows in set (0.00 sec) 














男 外 ， 如 果 是 “销售 人 额 (元 )” 这 种 包含 半角 括号 的 情况 ， 就 需要 使 用 " " 将 整个 内 容 括 起 来 ， 
即 “" 销售 额 (元 )"”， 否 则 就 会 发 生 错 误 "“， 这 一 点 需要 注意 。 

除了 可 以 像 上 面 那样 进行 乘法 或 除法 运算 ， 各 个 列 值 之 间 也 可 以 进行 计算 。 我 们 可 以 使 用 下 面 
的 命令 让 列 a 的 值 除 以 列 b 的 值 。 


SELECT a/b FROM 表 名 ; 


列 a 的 值 和 列 b 的 值 相 加 时 使 用 下 面 的 命令 。 


SELECT a+b FROM 表 和 名 ; 


当然 ， 这 都 是 在 同一 条 记录 的 各 列 值 之 间 进 行 的 计算 。 














中 使 用 全 角 括 号 时 ， 能 够 正常 执行 ， 不 会 发 生 错误 。 





译 者 注 
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8.2.2 ”使 用 函数 进行 计算 


1 一 一 一 一 一 一 一 
传人 数据 后 ， 函 数 会 执行 指定 的 处 理 并 返回 结果 。 例 如 ，AVG 这 个 函数 会 返回 传人 数据 的 平均 

值 。 函 数 名 的 后 面 要 加 上 () ，() 里 面 是 需要 处 理 的 数据 。 放 在 () 中 的 数据 称 为 参数 。 
虽然 上 面 提 到 的 函数 与 Excel 等 表 计 算 的 函数 基本 相同 ， 但 在 SQL 语句 中 ，( ) 中 指定 的 大 多 

是 列 名 。 例 如 ，AVG (x) 会 返回 列 x 中 数据 的 平均 值 。 由 于 列 x 的 值 和 记录 数 相对 应 ， 所 以 该 函数 





表示 列 x 中 所 有 记录 的 平均 值 。 





当 指 定 作为 函数 处 理 对 象 的 记录 时 ， 我 们 可 以 使 用 WHERE 设置 条 件 进行 提取 ， 也 可 以 使 用 
GROUP BY 对 记录 进行 分 组 计算 (一 8.6 节 )。 
下 面 我 们 来 计算 一 下 D 公司 2018 年 第 二 季度 的 平均 销售 额 。 试 着 显示 表 tb 的 列 sales 的 平 

















均值 。 


P 执行 内 容 
> 执行 前 ( 表 tb ) 


> 执行 后 




















。 | 时 示 平 均值 





















































empid sales month AVG (sales) 
A103 101 4 》 123.4000 
A102 54 5 

A104 181 4 

A101 184 4 

A103 i 5 

A101 300 5 

A102 205 6 

A104 98 5 

A103 和 多 6 

A107 BB 6 





?| 该 列 数 据 











操作 方法 
执行 下 面 的 命令 。 
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mysql> SELECT AVG (sales) FROM tb; 





+------------ 十 
| AvVG (sales) | 
+------------ 十 
| 123.4000 | 
全 二 二 有 且 总 党 本 全 二 十 


1 row in set (0.00 sec) 











可 以 看 到 销售 额 ( sales ) 的 平均 值 是 123.4。 


> 显示 总 和 
下 面 的 命令 用 于 显示 销售 额 总 和 ( SUM 函数 )。 


SELECT SUM(sales) FROM tb; 


mysql> SELECT SUM(sales) FROM tb; 








相 宇 汪汪 访 和 入 十 
| SUM(sales) | 
还 十 
| 1234 | 
| 十 


1 row in set (0.00 sec) 











销售 额 总 和 为 “1234”。 


P 显示 个 数 
下 面 的 命令 用 于 统计 记录 个 数 ( COUNT 函数 )。 


SELECT COUNT (sales) FROM tb; 


mysql> SELECT COUNT (sales) FROM tb; 





1 row in set (0.00 sec) 
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个 数 是 “10”"。 因 为 销售 额 总 和 是 “1234”， 而 个 数 是 “10”"， 所 以 可 以 得 知 平均 值 与 AVG (sales) 


返回 的 123.4 一 致 。 


MySQL 中 有 很 多 内 置 函 数 。 除 了 这 里 介绍 的 用 于 计算 平均 值 、 总 和 以 及 个 数 的 统计 函数 ， 还 
有 sin、cos 这 种 数学 处理 函数 ， 以 及 字符 串 函 数 、 日 期 和 时 间 函 数 等 。 本 书 仅 介绍 了 其 中 的 一 小 





部 分 。 


8.2.3 ”用 于 显示 各 种 信息 的 函数 
下 面 来 介绍 一 个 和 表 完 全 无 关 的 函数 。PI 是 用 于 返回 圆周 率 的 函数 。 试 着 执行 下 面 的 合 




















今 


O 


SELECT PI(); 


执行 结果 





mysql> SELECT PI (); 


1 row in set (0.11 sec) 











像 这 样 将 函数 和 SELECT 一 起 使 用 ， 就 能 够 显示 出 与 列 和 表 无 关 的 数据 。 
下 面 介 绍 的 函数 用 于 查看 当前 使 用 的 MySQL 环境 。 














区 显示 MySQL 服务 器 版 本 


SELECT VERSION(); 


区 显示 当前 使 用 的 数据 库 


SELECT DATABASE(); 


P 显示 当前 用 户 


SELECT USER(); 
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> 显示 由 参数 指定 的 字符 的 字符 编码 


SELECT CHARSET(' 这 个 字符 '); 





即使 不 需要 用 到 参数 ， 函 数 名 之 后 也 要 加 上 ( ) 。 省 略 了 ( ) 就 会 发 生 错 误 。 函 数 后 面 必 须 加 上 
() 是 很 多 编程 语言 的 共通 之 处 。 之 后 介绍 的 存储 过 程 (一 12.1 节 )、 存 储 函 数 (一 12.4 节 ) 和 
PHP 函数 (一 15.8.4 节 ) 也 都 需要 加 上 () 。 


计算 圆 的 面积 
如 果 使 用 保存 了 半径 的 列 r 的 值 计 算 面积 ， 就 可 以 将 语句 编写 为 “SELECT r * r * PI()FROM ...”o 






































8.2.4 连接 字符 串 

MySQL 中 还 有 很 多 用 于 处 理 字符 串 的 函数 。 例 如 在 连接 字符 串 的 时 候 ， 可 以 使 用 CONCAT 也 
数 。coNCAT 中 指定 的 字符 串 列 需要 使 用 “, ”进行 分 隔 。 比 如 在 连接 列 a、b 、e 的 字符 时 ， 就 需 
要 写成 CONCAT (a,b,c) 。 该 函数 还 可 以 直接 指定 字符 串 。 

例如 ， 员 工 信 息 表 tbl 的 列 empid 表示 员工 号 ， 列 name 表示 姓名 。 下 面试 着 显示 内 容 为 “ 员 
工 号 + 姓名 + 先生 ”的 字符 串 。 

连接 表 tbl 的 列 empid、 列 name 以 及 “先生 ”并 将 其 显示 出 来 。 


PP 执行 内 容 


区 执行 前 ( 表 tb1 ) > 执行 后 





CONCAT (empid,name,' 先生 ') 











连接 起 来 后 ， 加 上 
“先生 ”显示 出 来 
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操作 方法 


执行 下 面 的 命令 。 





RA101 佐 蕨 先生 
A102 高 桥 先 生 
A103 中 川 先生 









































5 rows in set (0.14 sec) 











当然 ，CONCAT 函数 也 可 以 连接 3 个 以 上 的 字符 串 。 男 外 , “先生” 是 字符 串 数据 ， 不 要 忘 
使 用 “'” 把 它 括 起 来 。 


人 8.2.5_ 字 符 串 操作 中 常用 的 画 数 


下 面 介绍 字符 串 操 作 中 一 些 常 用 的 函数 。 如 果 使 用 员工 信息 表 tb!1 实际 进行 测试 ， 就 能 很 好 地 
理解 这 些 函 数 的 作用 了 了 。 














> 从 右 取出 : RIGHT 函数 
下 面 的 命令 用 于 显示 列 empid 最 右边 的 2 个 字符 。 


SELECT RIGHT (empid,2) FROM tbl; 


在 列 empid 的 值 为 “A101” 的 情况 下 ， 显 示 为 “01”。 

















P 从 左 取 出 : LEFT 函数 
下 面 的 命令 用 于 显示 列 empid 最 左边 的 2 个 字符 。 


SELECT LEEFT (empid,2) FROM tb1)， 
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在 这 种 情况 下 ， 所 有 的 值 都 显示 为 “Al ”。 





P 从 第 x 个 字符 开始 截取 A 个 字符 : SUBSTRING 函数 
下 面 的 命令 用 于 从 列 empid 的 第 2 个 字符 开始 连续 显示 3 个 字符 。 


SELECT SUBSTRING (empid,2,3) FROM tbl; 


在 列 empid 的 值 为 “A101” 的 情况 下 ， 显 示 为 “101”。 


PP 重复 显示 : REPEAT 函数 
下 面 的 命令 用 于 重复 显示 字符 “.”， 其 重复 次 数 为 列 age 的 值 。 


SELECT REPEAT('.',age) FROM tb1; 


这 样 就 可 以 绘制 D 公司 员工 的 年 龄 简易 图 了 。 使 用 “了 ”等 特定 字符 会 变 得 更 加 有 趣 。 























mysql> SELECT REPEAT('.',age) FROM tbl: 
i + 

REPEAT('.',age) 
+------------------------------------------ 十 
汪汪 十 
5 rows in set (0.00 sec) 





PP 反 转 显示 : REVERSE 函数 
下 面 把 列 name 中 的 字符 串 倒 着 显示 出 来 。 














SELECT REVERSE (name) FROM tbl; 





这 样 ， 姓 名 就 会 倒 过 来 显示 。 
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| | 
| | 
| 川中 
| | 
| | 

















5 rows in set (0.00 sec) 











使 用 REVERSE 函数 能 够 找到 字符 顺序 颠倒 的 单词 。 
8.2.6 日 期 和 时 间 号 数 

处 理 日 期 和 时 间 的 函数 也 有 很 多 。 下 面 将 介绍 本 书 使 用 的 NOW 函数 。 

NOW 是 用 于 返回 当前 日 期 和 时 间 的 函数 。 如 果 想 自动 设置 执行 处 理 的 日 期 和 时 间 ， 可 以 使 用 
NOW () 。NOW () 会 返回 日 期 和 时 间 ， 所 以 最 好 将 输入 列 设置 为 DATETIME 类 型 。 第 5 部 分 介绍 的 
实用 公告 板 也 会 用 到 NOW () ， 大 家 先 记 住 这 个 函数 。 

作为 练习 ， 我 们 来 创建 表 t now。 将 表 t now 的 列 a 设置 为 INT 类型， 列 b 设置 为 DATETIME 
类 型 ， 并 给 列 a 添加 自动 连续 编号 功能 (一 6.8 节 )。 然 后 将 当前 的 日 期 和 时 间 保 存在 列 b 中 并 插入 
5 条 记录 。6.13 节 中 出 现 过 类 似 的 练习 题 。 


PP 执行 内 容 

























































































> 创建 表 t_now 
列 a b 
INT 
数据 类 型 | AUTO INCREMENT | DATETIME 
i 一 | 设 动 连续 编号 功能 
Be 设置 自动 连续 编号 功能 




















D4 


























> 执行 后 

a b 

1 输入 时 的 时 间 

2 输入 时 的 时 间 

3 输入 时 的 时 间 | 输入 输入 时 的 时 间 | 
4 输入 时 的 时 间 

5 输入 时 的 时 间 
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操作 方法 






































: INSERT INTO t now (b) VALUES (NOW() ) ; 


(3) 反复 执行 4 次 @)。 

















使 用 Enter 键 输入 后 ， 如 果 需 要 反复 输入 相同 的 数据 ， 可 以 先 按 1 键 ， 再 按 Enter 键 ， 反 复 执 
了 该 操作 。 如 果 在 列 b 中 输入 Now () ， 执 行 处 理 时 的 时 间 就 会 被 输入 进去 。 

我 们 来 确认 一 下 输入 的 记录 。 请 执行 “SELECT * FROM t_now;” 命 令 。 

| 执行 结果 | 


mysql> SELECT * FROM t now; 





一 人 




















| 
a + 
| 1 | 2018-05-27 14:52:50 | 
| 2 | 2018-05-27 14:52:54 | 
| 3 | 2018-05-27 14:52:55 | 
| 4 | 2018-05-27 14:52:56 | 
| 5 | 2018-05-27 14:54:27 | 
于 时 中 


5 rows in set (0.00 sec) 

















由 于 给 列 a 设置 了 自动 连续 编号 功能 ， 所 以 从 1 开始 的 连续 编号 也 会 输入 进去 。 











8.3 设置 条 件 进行 显示 


在 本 书 中 ,我们 处 理 的 表 大 约 只 有 10 条 记录 。 因 此 在 执行 SELECT 时 ， 所 有 结果 都 能 立刻 显 
示 出 来 。 但 是 在 实际 工作 中 ， 我 们 处 理 的 表 可 能 会 有 成 千 上 万 条 记录 。 

显示 大 量 无 意义 的 记录 只 会 降低 工作 效率 。 所 以 大 家 要 掌握 限制 显示 的 记录 数 的 方法 。 
8.3.1 确定 记录 数 并 显示 

我 们 可 以 使 用 LIMIT 来 限制 要 显示 的 记录 数 。 
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限制 要 显示 的 记录 数 


eeeeeeeoeooseeeeooooooooooooooooooeoooooeosoooooeoooooooeoooooeooooooooooooooooooooooooooooooeoooooooooooooeooooooooooooooooooooooooooooooooe 


我 们 试 着 让 销售 信息 表 tb 仅 显 示 3 条 记录 。 


区 执行 内 容 


> 执行 前 ( 表 tb ) 








empid 

































































A101 184 4 

A103 17 5 

A101 300 5 这 3 条 记录 
A102 205 6 

A104 93 5 

A103 12 6 

A107 87 6 

操作 方 ; 

执行 下 面 的 命令 。 


mysql> SELECT * FROM tb LIMIT 3; 


+------- +------- +------- + 





| empid | sales | month | 


于 和 再 入 和 和 汕 和 和 二 本 
| A1l03 | 101 | 4 | 
| A1l02 | 54 | 5 | 
| Al04 | 181 | 4 | 
汪 二 下 天生 十 








3 rows in set (0.00 sec) 





这 样 就 只 会 显示 3 条 记录 了 。 
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数据 库 的 处 理 速度 和 各 种 因素 有 关 。 在 很 多 情况 下 只 能 通过 经 验 来 预测 处 理 需 要 花费 的 时 间 。 
然而 ， 花 费 的 时 间 只 有 尝试 之 后 才能 知道 。 而 使 用 ZIMIT 通常 可 以 使 结果 更 快 地 显示 出 来 。 因 此 ， 
对 于 需要 花费 一 定时 间 的 处 理 ， 最 好 先 使 用 LIMIT 显示 几 行 看 看 。 

另外 ,“ 将 第 O 〇 行 到 第 x 行 的 内 容 显示 出 来 ”的 显示 方法 将 在 8.5 节 进 行 介绍 。 


























下 面 将 介绍 使 用 WHERE 提取 记录 的 方法 。 

我 们 可 以 使 用 WHERE 设置 条 件 并 取出 与 条 件 相 匹配 的 记录 。 能 否 顺 利 提取 记录 可 以 说 取决 于 
如 何 使 用 WHERE。 

如 果 单 纯 地 执行 表示 删除 的 DELETE 或 表示 更 新 的 UPDATE (一 9.1 节 )， 所 有 记录 都 会 被 删除 
或 者 更 新 。 如 果 使 用 WHERE 设置 条 件 ， 就 可 以 将 需要 用 到 的 记录 作为 删除 或 更 新 的 对 象 了 。 

要 想 将 提取 的 内 容 限 定 为 符合 某 一 条 件 的 记录 ， 就 需要 按照 下 面 的 方法 使 用 WHERE。 


仅 显示 符合 条 件 的 记录 





















































关于 “条 件 ” 的 写法 ， 例 如 当 列 a 的 值 大 于 等 于 10 时 ,“ 条 件 ” 可 以 写成 a>=10。 
在 销售 信息 表 了 b 中 ， 列 sales 表示 销售 额 。 下 面 我 们 试 着 只 显示 列 sales 中 值 大 于 等 于 100 的 记录 。 








PP 执行 内 容 



















































































> 执行 前 ( 表 tb ) > 执行 后 

empid sales month empid sales month 

A103 101 4 A103 101 4 

A102 54 5 》 A104 | 181 4 a 
A104 181 4 A101 184 4 并 显示 
A101 184 4 A101 300 5 

A103 全 5 A102 205 6 

A101 300 

A102 205 6 

A104 93 5 

A103 12 6 

A107 87 6 



































该 列 中 值 大 于 等 于 100 的 记录 
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操作 方法 


执行 下 面 的 命令 。 

















执行 结果 

mysql> SELECT * FROM tb WHERE sales>=100; 

+------- +------- +------- 十 
empid sales month 

Fs 于 = 三 = Es + 
A103 101 4 
Al04 181 4 
Al01 184 4 
Al01 300 Ss 
A1l02 205 6 

+------- +------- +------- 十 








5 rows in set (0.05 sec) 





@ 8.3.3 ”比较 运算 符 


前 面 我 们 使 用 了 符号 “>=” 来 设置 “大 于 等 于 x x” 的 条 件 ， 这 种 符号 称 为 比较 运算 符 。 
MySQL 中 常用 的 比较 运算 符 如 表 8-2 所 示 。 




















表 8-2 常用 的 比较 运算 符 






































比较 运算 符 含义 

等 于 

> 大 于 

、- 大 于 等 于 

< 小 于 

小 于 等 于 

<> 不 等 于 

O IN x 〇 在 x 列表 中 

O NOT IN x O 〇 不 在 x 列表 中 

O BETWEEN x AND xx OO 在 x 到 x x 之 间 
O NOT BETWEEN x AND xx 〇 不 在 x 到 x x 之 间 
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12 


9 


下 面 介绍 儿 个 使 用 数值 作为 WHERE 条 件 的 示例 。 无 论 执行 多 少 次 SELECT 命令 都 不 会 引起 问 


题 ， 所 以 大 家 可 以 使 用 销售 信息 表 tb 进行 各 种 各 样 的 试验 。 


P 列 sales 的 值 小 于 50 


SELECT * FROM tb WHERE sales<50; 





mysql> SELECT * FROM tb WHERE sales<50; 
+------- +------- +------- 十 
| empid | sales | month | 


2 rows in set (0.00 sec) 











PP 列 month 的 值 不 等 于 4 


SELECT * FROM tb WHERE month<>47 


P 列 sales 的 值 在 50 到 100 之 间 ( 大 于 等 于 50、 小 于 等 于 100 ) 


SELECT * FROM tb WHERE sales BETWEEN 50 AND 100; 


P 列 sales 的 值 不 在 50 到 200 之 间 ( 小 于 50 或 大 于 200 ) 


SELECT * FROM tb WHERE sales NOT BETWEEN 50 AND 200; 


区 列 month 的 值 等 于 5 或 者 6 


SELECT * FROM tb WHERE month IN (5,6); 
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执行 结果 

mysql> SELECT * FROM tb WHERE month IN (5,6); 

+------- +------- +------- 十 
empid sales month 

+------- +------- +------- 十 
A1l02 54 5 
A1l03 17 5 
Al01 300 5 
A102 205 6 
Al04 93 全 
Al103 上 人 6 
Al107 87 6 

+------- +------- +------- 十 

7 rows in set (0.00 sec) 














我 们 可 以 利用 10.5 节 介 绍 的 子 查 询 先 用 SELECT 提取 记录 ， 然 后 提取 结果 中 包含 的 记录 。 在 
这 两 个 阶段 的 提取 操作 中 ，IN 也 会 起 到 非常 关键 的 作用 。 











前 面 我 们 学 习 了 数值 条 件 的 设置 方法 ， 这 次 将 学 习 字 符 串 相关 的 设置 方法 。 假 设 在 表 tb 中 ， 
列 empid 是 字符 串 类 型 。 如 果 想 显示 empid 为 A101 的 记录 ， 和 数值 一 样 使 用 “=” 设 置 条 件 即 可 。 


SELECT * FROM tb WHERE empid='A101'; 


在 这 种 情况 下 ， 列 empid 的 内 容 必须 与 “A101” 完 全 匹配 。 因 此 ,“A0101” 或 “AA101” 的 
记录 不 会 被 提取 出 来 。 














区 LIKE : 模糊 查询 

当 提 到 某 个 人 的 名 字 时 ， 我 们 有 时 会 采用 一 种 含糊 不 清 的 表达 。“ 是 中 川 还 是 川中 ” 总 之 名 字 
里 有 一 个 川 字 。” 查 询 数据 库 时 也 会 出 现 这 样 的 情况 。 

比如 既 想 提取 A101， 又 想 提取 A102， 这 时 就 可 以 执行 模糊 查询 。 如 果 想 使 用 “包含 x x 字 
符 ” 这 种 含糊 不 清 的 条 件 执行 查询 ， 就 需要 用 到 LIKE。 例 如 包含 “ 啊 ” 这 个 字符 的 条 件 要 写成 
LIKE' 啊 '。 前 面 的 SQL 语句 如 果 按 照 下 面 的 方式 书写 ， 结 果 也 不 会 发 生 任何 改变 。 


SELECT * FROM tb WHERE empid LIKE 'Al01'; 


LIKE 可 以 把 包含 某 字符 串 的 所 有 内 容 当 成 查询 对 象 。 在 这 种 情况 下 ， 需 要 用 到 “%” 和 “_” 
等 通配符 ( 见 表 8-3 )。 当 设置 条 件 时 ,“%” 代 表 任 意 字 符 串 ，” ”代表 任意 一 个 字符 。 
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表 8-3 字符 串 的 通配符 及 其 使 用 示例 


通配符 含义 
s 任意 字符 


















































































































































指定 的 字符 串 符合 的 例子 

gs 县 绮 玉 县 、 非 常 好 的 县 、 废 藩 置 县 、 县 
福 有 福井 县 、 福 岛 、 福 

长 ”县 和 野 县 、 长 崎 县 、 长 海 县 

县 多 包含 县 就 可 以 ， 县 、 县 名 、 长 海 县 










































































例如 对 于 表 tb 的 列 empid， 即 使 不 特意 写成 empid ='A101'， 也 可 以 像 下 面 这 样 以 “最 后 是 
1” 为 条 件 来 提取 A101 的 记录 。 


ey 


“%1” 表 示 任 意 字 符 串 +1。 下 面 我 们 试 着 显示 表 tb1 的 列 name 中 包含 字符 “ 川 ” 的 记录 。 




















P> 执行 内 容 








































































































> 执行 前 ( 表 tb1 ) 区 执行 后 
empid name age empid name age 
A101 佐 茧 40 》 A103 中 川 20 eo sg 
A102 高 桥 28 
A103 | 中川 20 
A104 渡 边 23 
A105 西 泽 35 
i 的 记录 














操作 方法 
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执行 结果 





mysql> SELECT * FROM tbl WHERE name LIKE '%)l|%'; 
+------- +------ +------ 十 
| empid | name | age | 





1 row in set (0.03 sec) 








如 果 表 中 有 “ 川 "“ 川 田 ”或 者 “小 副 川 "， 也 会 一 并 提取 出 来 。 我 们 可 以 通过 name LIKE 
, 川 ! 进行 前 方 _ 致 检索， 通过 name LIKE '% 川 ， 进行 后 方 - 臻 检索, 通过 name LIKE '% 川 
进行 部 分 一 致 检索 。 





区 提取 不 包含 指定 字符 串 的 记录 


另外 ， 提 取 不 包含 某 字 符 串 的 记录 时 需要 使 用 NOT LIKE 命令 。 执 行 下 面 的 命令 可 以 提取 表 
tbl 的 列 name 中 不 以 字符 “ 优 ” 开 头 的 记录 。 


SELECT * FROM tbl WHERE name NOT LIKE ' 优 %'; 


局 8.3.5 使 用 NULL 作 为 条 件 


NULL 表示 空 值 。 如 果 没 有 向 列 中 输入 数据 ， 也 没有 给 列 设 置 默认 值 ， 就 会 输入 NULL。 对 于 和 员 
工 信 息 表 tb1 内 容 相 同 的 表 tblH， 假 如 我 们 仅 向 姓名 列 name 中 输入 数据 ， 其 他 列 中 就 会 输入 NULL。 


INSERT INTO tblH (name) VALUES (' 仅 仅 是 姓名 ': ) ; 

































































执行 结果 

mysql> SELECT * FROM tblH: 

+------- +------------ +------ 十 
empid name age 

+------- +------------ +------ 十 
A1l01 佐 腾 40 
A102 高 28 
A103 中 川 20 
RAR104 渡 边 23 
A1l05 泽 35 
NULL 仅仅 是 姓名 NULL 

+------- +------------ +------ 十 

6 rows in set (0.00 sec) 














> 当 列 值 为 NULL 时 
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提取 列 值 为 NULL 的 记录 时 需要 使 用 IS NULL。 例如 ， 下 面 的 命令 用 于 提取 表 tb1H 中 列 age 





为 NULL 的 记录 。 


SELECT * FROM tblLH WHERE age IS NULL; 


> ” 当 列 值 非 NULL 时 





反之 ,提取 列 值 不 是 NULL 的 记录 时 需要 使 用 Is NOT NULL。 下 面 的 命令 用 于 提取 表 tb1H 中 


列 age 不 为 NULL 的 记录 。 


SELECT * FROM tblH WHERE age IS NOT NULL; 





需要 注意 的 是 ， 当 提取 值 为 NULL 的 记录 时 ， 即 使 使 有 





相应 的 记录 。 


删除 多 余 的 记录 





月 WHERE age = NULL， 也 无 法 提取 出 


例如 ， 执行 “SELECT empid FROM tb;” 会 重复 提取 3 个 “A103”。 














10 rows in set (0.03 sec) 





























表 tb 中 有 10 条 记录 ， 所 以 执行 结果 才 会 变 成 上 面 那样 。 但 是 从 用 户 的 



























































度 来 说 ， 很 多 时 候 显示 重复 的 记录 





并 没有 什么 意义 。 此 时 如 果 像 下 面 这 样 给 命令 加 上 DISTINCT， 就 可 以 删除 多 余 的 记录 了 。 
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SELECT DISTINCT empid FROM tb; 











mysql> SELECT DISTINCT empid FROM tb; 
+------- + 
empid 
+------- 十 
A103 
A1l102 
Al04 
Al01 
A1L07 
+------- 十 
SEOSSE OUS SS 











8.4 指定 多 个 条 件 进 行 选择 


代表 “OO 和 x x” 的 AND， 以 及 代表 “OO 或 x x” 的 oOR， 是 很 多 编程 语言 拥有 的 逻辑 运 
算 符 。 我 们 试 着 将 WHERE 与 AND 或 OR 放 到 一 起 使 用 ， 以 此 来 设置 多 个 条 件 。 





C 8.4.1 使 用 AND 


如 果 想 设置 “OO 和 x x” 这 样 的 条 件 ， 就 可 以 使 用 AND 将 多 个 条 件 连 接 起 来 。 例 如 ， 在 8.3.3 
节 介 绍 “ 列 sales 的 值 大 于 等 于 50、 小 于 等 于 100” 的 示例 时 ， 我 们 提 到 了 “sELECT * FROM tb 
WHERE sales BETWEEN 50 AND 100;”。 如 果 使 用 AND 编写 该 条 件 ， 就 是 下 面 这 种 形式 。 


SELECT * FROM tb WHERE sales>=50 AND sales<=100; 


在 使 用 AND 的 情况 下 ， 可 以 给 不 同 的 列 设置 多 个 条 件 。 
表 也 显示 了 各 员工 号 所 对 应 的 月 销售 额 。 那 么 ， 我们 应 该 如 何 查 询 满足 以 下 条 件 的 记录 呢 ? 











2 员工 号 为 A101 的 员工 4 月 份 的 销售 额 





我 们 可 以 试 着 使 用 LIKE (一 8.3.4 节 ) 执行 模糊 查询 。 显 示 的 记录 需要 满足 列 empid 的 值 的 最 
后 为 “1” 且 列 month 的 值 为 “4” 的 条 件 。 
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PP 执行 内 容 











































































































> 执行 前 ( 表 tb ) > 执行 后 
empid sales month empid sales month 
A103 101 4 小 A101 184 4 e 一 一 提取 记录 并 
显示 出 来 
A102 54 5 
A104 181 4 
A103 17 5 
A101 300 5 
A102 205 6 
A104 93 5 
A103 12 6 
A107 87 6 
9 9 
列 值 的 最 后 是 “1” 列 值 为 “4” 
操作 方法 





执行 下 面 的 命令 。 

















SELECT * FROM tb WHERE empid LIKE '$%1' AND month=4; 








1 row in set (0.00 sec) 











可 以 看 到 员工 号 “Al101” 的 员工 4 月份 的 销售 额 是 184 万 元 。 
在 这 个 示例 中 , 使 用 empiq LIKE 'A101' 或 者 empid='A101' 都 可 以 将 记录 提取 出 来 ,不 
过 我 们 使 用 了 后 方 一 致 检索 (一 8.3.4 节 ) 的 方法 。 








时 





C 8.4.2 使 用 OR 


下 面 我 们 要 设置 “OO 或 者 x x ”的 条 件 。 在 这 种 情况 下 需要 使 用 OR。 在 8.3.3 市 介 绍 的 “ 列 
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sales 的 值 小 于 50 或 大 于 200” 的 示例 中 ， 我 们 使 用 了 “SELECT * FROM tb WHERE sales 
NOT BETWEEN 50 AND 200;”。 同 样 ， 该 条 件 也 可 以 使 用 oR 来 处 理 。 
下 面试 着 显示 表 tb 中 列 sales 的 值 小 于 50 或 者 大 于 200 的 记录 。 




















PP 执行 内 容 


































































































> 执行 前 ( 表 tb ) 区 执行 后 
empid Sales month empid sales month 
A103 101 4 A103 17 5 
A102 54 5 A101 300 5 提取 记录 
A104 181 4 A102 205 6 并 显示 
A101 184 4 A103 12 6 
A103 山区 5 
A101 300 5 
A102 2205 6 
A104 93 5 
A103 le 6 
A107 87 6 
列 值 小 于 50 La 200 的 记录 

















操作 方法 


执行 下 面 的 命令 。 





mysql> SELECT * FROM tb WHERE sales<50 OR sales>200; 
+------- +------- +------- + 


| empid | sales | month | 


nt 起 ee 中 
| A103 | 17 | sl 
| A1l01 | 300 | s | 
| A102 | 205 | 6 | 
| A103 | 革 倒 韦 | 6 | 
二 和 ese 下 





4 rows in set (0.00 sec) 
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C 8.4.3 使 用 多 个 AND 或 OR 


我 们 可 以 设置 任意 次 数 的 AND 和 OR。 下 面 是 同时 使 用 AND 和 oR 进行 处 理 的 示例 。 
2 “empid 是 'A101'” 且 “month 是 4”, 或 者 “sales 大 于 等 于 200” 


也 就 是 说 ， 要 在 8.4.1 节 人 处 理 的 “员工 号 为 A101 的 员工 4 月 份 的 销售 额 ” 的 基础 上 添加 “销售 
额 大 于 等 于 200 万 元 ”的 记录 。 对 此 ， 我 们 可 以 使 用 下 面 的 语句 将 条 件 连接 起 来 。 





SELECT * FROM tb WHERE empid LIKE '%]1' AND month=4 OR sales>=200; 





mysql> SELECT * FROM tb WHERE empid LIKE '%1' AND month=4 OR sales>=200; 
+------- +------- +------- 十 





| empid | sales | month | 


和 和 Pe + 
| Alol | 184 | 4 | 
| A1l01 | 300 | 5 | 
| Alo2 | 205 | 6 | 
圳 和 二 en + 


3 rows in set (0.00 sec) 











可 以 看 到 ， 员工 号 为 A101 的 员工 4 月 份 的 记录 中 增加 了 2 条 sales 大 于 200 的 记录 。 即 使 像 下 
面 这 样 修改 条 件 的 设置 顺序 ， 处 理 也 不 会 发 生 任何 改变 。 


SELECT * FROM tb WHERE sales>=200 OR empid LIKE '%1' AND month=4; 





也 就 是 说 ， 无 论 条 件 的 设置 顺序 如 何 ， 都 会 先 处 理 empiqd LIKE '%1' AND month=4， 然 
后 再 用 OR 添加 sales 大 于 等 于 200 的 记录 。 请 记 住 下 面 的 规则 。 





DS 当 AND 和 OR 混合 使 用 时 ， 会 优先 处 理 AND 





那么 ， 如 果 想 进行 下 面 的 处 理 ， 大 家 知道 该 怎么 做 吗 ? 











2 “sales 大 于 等 于 200， 或 者 empid 为 'A101'” 且 “month 是 4” 


该 处 理 表示 先 提取 满足 销售 额 大 于 等 于 200 万 元 ,或 者 员工 号 为 A101 这 一 条 件 的 记录 ， 然 后 
再 从 结果 中 提取 满足 4 月份 这 个 条 件 的 记录 。 我 们 想 优 先 处 理 的 是 OR 的 部 分 ， 但 是 直接 执行 会 先 
处 理 AND 的 部 分 。 

所 以 ， 在 这 种 情况 下 ， 我 们 需要 用 () 把 想 要 优先 处 理 的 内 容 括 起 来 。 























Ey 
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执行 结果 





mysql> SELECT * FROM tb WHERE (sales>=200 OR empid LIKE '$1')AND month=4; 
+------- +------- +------- 十 
| empid | sales | month | 


| A1l01 | 184 | 4 | 
+------- +------- +------- + 


1 row in set (0.06 sec) 











通过 表 tb 可 以 得 知 ，4 月 份 没有 大 于 等 于 200 的 记录 ， 所 以 只 有 员工 A101 的 4 月 份 的 记录 保 
留 了 下 来 。 
® 8.4.4 使 用 CASE WHEN 

还 有 一 种 “根据 条 件 改变 输入 值 ” 的 高 级 方法 。 例 如 ， 大 于 等 于 80 分 输入 “ 优 ?， 大 于 等 于 60 
分 小 于 80 分 输入 “ 良 ”， 大 于 等 于 40 分 小 于 60 分 输入 “及 格 ”， 除 此 之 外 全 部 输入 “不 及 格 ”， 对 
于 这 种 处 理 ， 我 们 可 以 使 用 cASE WHEN。 


根据 条 件 改 变 并 显示 值 


WHEN 条 件 1 THEN 显示 的 值 
WHEN 条 件 2 ”THEN 显示 的 值 
WHEN 条 件 3 THEN 显示 的 值 














: ELSE 不 满足 所 有 条 件 时 的 值 


当 使 用 sELECT 命令 显示 列 值 时 ， 我 们 可 以 使 用 上 述 语句 来 记述 列 的 内 容 。 如 果 需 要 显示 多 个 
列 ， 像 之 前 一 样 使 用 “, ”进行 分 隔 即 可 。 
下 面 我 们 使 用 销售 信息 表 tb 对 销售 额 进 行 如 下 评价 。 





























2 当 销 售 额 ( sales ) 大 于 等 于 100 时 为 “高 "， 大 于 等 于 50 小 于 100 时 为 “中 等 "， 否 则 为 “ 低 ” 


用 于 显示 上 述 内 容 的 SQL 语句 如 下 所 示 。 


SELECT 
CRSBE 
WHEN sales>=100 THEN ' 高 ' 


WHEN sales>=50 THEN ' 中 等 ' 
ELSE ' 低 ' 

END 

FROM tb; 
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把 所 有 内 容 放 在 一 行 记 述 看 起 来 有 些 宛 长 ， 所 以 这 里 执行 了 换行 并 再 添加 了 缩 进 (一 7.9 节 )。 
我 们 先 记 住 这 个 “cASE WHEN ...” 的 用 法 。 使 用 表 tb， 通 过 cASE 根据 列 sales 的 值 显示 


“高”“ 中 等 ”和 “ 低 ”。 
但 是 ， 直 接 执行 这 个 命令 ， 会 出 现 预 想 之 外 的 结 


mysql> SELECT 
-> CASE 
-> WHEN sales>=100 THEN ' 高 ' 
-> WHEN sales>=50 THEN ' 中 等 ' 
3 ELSE ' 低 ' 




















WHEN sales>=100 THEN ' 高 ' 
WHEN sales>=50 THEN ' 中 等 ' 
ELSE ' 低 ' 

















10 rows in set (0.05 sec) 



































显示 出 来 的 内 容 让 人 有 些 难 以 理解 。 开 头 出 现 了 乱糟糟 的 “CasE WHEN ...” 字 符 串 ,导致 显 
示 的 内 容 过 于 宛 长 。 

执行 SELECT 命令 会 先 显示 出 列 名 等 表示 项 目的 描述 。 在 前 面 的 例子 中 ， 列 名 都 是 empid 或 
sales 这 种 简短 的 形式 ， 但 是 这 次 显示 的 列 名 是 “CAsE WHEN .… END”。 

这 种 列 名 会 对 理解 造成 障碍 ， 所 以 我 们 试 着 给 CASE ... END 的 部 分 加 上 别名 “评价 ”。 不 过 ， 
光 显 示 评 价 并 不 能 看 出 它 表 示 的 是 什么 ， 因 此 我 们 也 让 列 empid 和 列 sales 显示 出 来 。 























140 | 第 8 章 使 用 各 种 条 件 进 行 提取 


PP 执行 内 容 




















































































































































































































> 执行 前 ( 表 tb ) > 执行 后 

empid | sales month empid Sales [评价 ”一 别名 “评价 

A103 101 4 大 于 等 于 100 A103 NON 高 Pe < 加 

A104 181 4 A104 181 高 

A101 184 4 小 A101 184 高 

A103 17 5 A103 17 低 

A101 300 5 A101 300 高 

A102 O05 6 A102 205 高 

A103 12 6 除 此 之 外 A103 但 低 - 一 
A107 E 6 除 此 之 外 均 显示 为 “ 低 ” 








操作 方法 
执行 下 面 的 命令 。 


SELECT empid,sales, 
CASE 
WHEN sales>=100 THEN ' 高 ' 
WHEN sales>=50 THEN ' 中 等 ' 
ELSE ' 低 ' 
END AS 评价 
FROM tb; 














mysql> SELECT empid,sales, 
入 CASE 
-> WHEN sales>=100 THEN ' 高 ' 
-> WHEN sales>=50 THEN ' 中 等 ' 
-> ELSE ' 低 ' 
= END AS 评价 
-> FROM tb; 

+------- +------- +------ + 

| empiqd | sales | 评价 | 

+------- +------- +------ + 

| Alo3 | 101 | 高 | 
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A102 54 中 等 
R104 加 高 
A1l01 184 高 
RA103 7 低 
R101 300 高 
A102 205 高 
A104 93 | 中 等 
RA103 12 | 低 
A107 87 | 中 等 
和 灶 入 和 3 + 
10 rows in set (0.00 sec) 











为 了 给 CASE ... END 设置 别名 ， 我 们 加 上 了 “as 评价 ”这 样 的 记述 。 要 显示 的 项 目 也 像 
“empid, sales,CASE ...” 这 样 用 “,” 分 隔 开 来 。 
如 果 把 命令 写 在 一 行 中 ， 就 是 下 面 这 样 。 




















SELECT empid,sales,CASE WHEN sales>=100 THEN ' 高 ' WHEN sales>=50 THEN ' 中 等 ' 


ELSE ' 低 ' END AS 评价 FROM tb; 








对 于 使 用 了 cASE WHEN 的 SQL 语句 ， 还 是 进行 换行 和 缩 进 更 有 助 于 理解 。 


8.5 排序 


人 8.5.1 按 升序 排序 并 显示 


通过 SELECT 命令 显示 的 记录 顺序 是 不 规则 的 。 如 果 记 录 的 数量 像 本 书 介 绍 的 示例 一 样 不 超过 
10 条 ， 就 不 会 出 现 什么 问题 。 但 如 果 记 录 有 几 千 条 、 几 万 条 ， 显 示 顺 序 就 非常 重要 了 。 除 非 指定 了 
显示 顺序 ， 和 否则 不 管 结果 如 何 ， 都 没有 理由 抱怨 数据 库 。 

我 们 可 以 使 用 ORDER BY 按 指定 的 列 值 顺序 显示 记录 。 


按 升序 显示 记录 





eeeeeeeeeeeeeeeeeoesooeeoeoeooeooeoeoooeoeoeoooooeoeoooooeoooooeeeooooeooooooeeoooooooooooooooooooooooooeeooooooooeoooooooeoooooooeooeooeooooooooeoo， 


在 这 种 情况 下 ， 记 录 会 按照 从 小 到 大 的 顺序 排列 ， 即 按 升序 排列 。 如 果 让 表 了 b 按照 销售 额 从 
低 到 高 的 顺序 显示 记录 ， 就 需要 按照 下 面 的 方式 进行 操作 。 
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区 执行 内 容 















































> 执行 前 ( 表 tb ) > 执行 后 

empid empid sales month 

A103 A103 

A102 A103 

A104 A102 

A101 A107 

A103 了 A104 显示 所 
A101 A103 有 记录 
A102 A104 

A104 A101 

A103 A102 

A107 A101 

按 升序 排列 该 列 的 值 
操作 方法 




















mysql> SELECT * FROM tb ORDER BY sales; 
+------- +------- +------- 十 
empid sales month 
+------- +------- +------- 十 
A103 12 6 
A1l03 i 与 
A1l02 54 Ss 
A1l07 87 6 
Al04 33 号 
A103 101 4 
Al04 181 4 
Al01 184 4 
A1l02 205 6 
A1l01 300 5 
+------- +------- +------- 十 
10 rows in set (0.05 sec) 
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在 什么 都 不 指定 的 情况 下 ， 记 录 会 按 升序 排列 。 但 如 果 想 明确 指定 按 升序 排列 ， 就 需要 像 下 卫 
这 样 给 命令 加 上 Asc。 执 行 结果 是 一 样 的 。 


SELECT * FROM tb ORDER BY sales ASC; 


S 8.5.2 ” 按 降序 排序 并 显示 

按 降序 排序 的 时 候 ， 需 要 给 命令 加 上 DESC。 需 要 注意 的 是 ，DESC 和 Asc 需要 写 在 “ORDER 
BY 列 名 ”的 后 面 。 

试 着 对 销售 信息 表 tb 进行 排序 并 设置 记录 数 (一 8.3.1 节 )， 然 后 将 结果 显示 出 来 。 试 着 在 表 
tb 中 按照 列 sales 的 值 从 大 到 小 的 顺序 显示 前 5 条 记录 。 

















PP 执行 内 容 


















































































































> 执行 前 ( 表 tb ) > 执行 后 

empid sales month empid sales month 

A103 101 4 A101 300 5 

A102 54 5 A102 205 6 

A104 181 4 A101 184 4 提取 记录 
A101 184 4 A104 181 4 0 
A103 17 5 A103 101 4 

A101 300 5 

A102 205 6 

A104 93 5 

A103 12 6 

A107 87 6 

名 的 记录 
操作 方法 
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执行 结果 
mysql> SELECT * FROM tb ORDER BY sales DESC LIMIT 5; 
= 六 写字 二 入伍 池 是 二 汪 二 二 二 < 十 

empid sales month 
村 于 和 = 十 

Al01 300 Ss 

A1l02 205 6 

Al01 184 4 

Al04 181 4 

Al03 101 4 
Pn 本 ER ES 十 


5 rows in set (0.05 sec) 














使 用 ORDER 进行 排序 ， 除 了 可 以 按 顺 序 显示 记录 以 外 ,在 设置 删除 或 更 新 等 条 件 的 时 候 也 非 
常 有 用 。 比 如 ， 处理“ 排序 后 删除 最 后 面 的 10 条 记录 ”( 一 9.3.3 节 ) 时 就 非常 方便 。 

















在 按 顺 序 显 示 记 录 的 情况 下 ， 如 果 能 指定 记录 的 显示 范围 就 会 非常 方便 。 在 8.3.1 节 中 ， 我 们 
使 用 LIMIT 来 限制 显示 的 记录 数 。 当 时 只 是 选择 了 3 条 记录 显示 出 来 ， 而 使 用 OFFSET 可 以 进 一 
步 指定 显 示 的 范 目 


指定 范围 并 显示 












































Cy 





o 




















“开始 显示 记录 的 移 位 数 ” 是 指定 “移动 多 少 位 后 开始 显示 记录 ”的 数字 。 如 果 设 置 为 
OFFSET 3， 则 表示 “从 第 1 条 记录 开始 移动 3 位 后 ， 从 第 4 条 记录 开始 显示 ”。 

对 于 销售 信息 表 tb， 我 们 按照 销售 额 ( 列 sales ) 从 高 到 低 的 顺序 将 排 在 第 4 位 和 第 5 位 的 记录 
显示 出 来 。 






































PP 执行 内 容 
> 执行 前 ( 表 tb ) 


























区 执行 后 

empid sales month 
A104 181 4 
A103 101 4 








PF 








提取 记录 并 








显示 4H 


来 










































































empid sales month 
A103 on 4 
A102 54 5 
A104 181 4 
A101 184 4 
A103 17 5 
A101 300 5 
A102 205 6 
A104 93 5 
A103 12 6 
A107 87 6 
0 质 序 排列 ， 
排 在 第 4 位 和 第 5 位 的 记录 








操作 方法 





执行 下 面 的 命令 。 




















2 rows in set 





mysql> SELECT * 
+------- +------- +------- + 
| empid | sales 


(0.00 sec) 


| month | 


FROM tb ORDER BY sales DESC LIMIT 2 OFFSET 3; 








146 | 第 8 章 使 用 各 种 条 件 进行 提取 
8.6 分 组 显示 


a8.6.1 分 组 显示 
在 表 tb 中 ，empid 为 “A101” 的 记录 有 多 个 。 我 们 可 以 让 同属 “A101” 的 多 条 记录 组 成 一 个 
组 合 。 这 样 就 能 以 组 为 单位 计算 该 组 记录 的 总 和 或 平均 值 了 。 
分 组 后 处 理 起 来 似乎 变 得 更 加 方便 了 ， 但 是 在 数据 库 中 ， 当 前 的 处 理 对 象 却 变 得 模糊 起 来 ， 所 
以 我 们 要 时 刻 了 解 当前 的 处 理 对 象 是 谁 。 
分 组 时 需要 使 用 GROUP BY 命令 。 


分 组 显示 


eeeeeeeeeeeeeeeeeoeeoeoeooeeooooooooooooooooooooeoooooooooooooooeoooooeooooooooooooooooooooooooooooooooooooeoooooooooooooooooooooooooooooooee 





























只 是 分 组 显示 的 话 并 没有 什么 意义 ， 不 过 我 们 还 是 先 试 着 按照 列 empid 进行 分 组 ， 看 看 会 显示 
出 什么 样 的 结果 。 
下 面 的 命令 用 于 显示 按照 列 empid 分 组 后 的 记录 。 


SELECT * FROM tb GROUP BY empid; 






































mysql> SELECT * FROM tb GROUP BY empid; 
+------- +------- +------- 十 
empid sales month 
+------- +------- +------- 十 
Al01 184 4 
A1l02 54 5 
A1l03 101 4 
Al04 181 4 
A107 87 6 
下 再 SEE 机 SEE 十 


5 rows in set (0.16 sec) 











表示 员工 号 的 empid 每 种 只 显示 了 一 个 值 。GROUP BY empid 使 每 组 empid 都 被 执行 了 
SELECT。 通 过 这 个 处 理 ， 我 们 可 以 知道 表 tb 的 列 empid 中 存在 Al101 、Al102 等 5 种 数据 。 

因为 要 按 组 进行 处 理 ， 所 以 我 们 姑且 可 以 认为 属于 该 组 的 记录 是 被 随机 选择 的 。 虽 然 “Al101” 
的 记录 对 应 的 “sales” 是 “184”， 但 这 只 是 随机 选 出 来 的 结果 。 一 定 要 记 住处 理 对 象 是 “同一 组 中 
的 所 有 记录 ”。 
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S 8.6.2 ”计算 各 组 的 记录 数 


ss 即 每 个 员工 号 有 多少 条 记录 。 计 算 个 数 需 要 使 用 COUNT 函数 
(一 8.2.2 节 )。COUNT (x) 用 于 计算 除 NULL 以 外 列 x 的 值 的 个 数 。 

前 面 我 们 提 到 过 ， es “同一 组 中 的 所 有 记录 ”。 例 如， 在 empid 为 “A101” 
的 组 中 ， 仅 显示 了 sales 为 “184” 的 记录 ,但 其 实 属于 “A101” 的 所 有 列 sales 的 值 都 是 处 理 
对 象 。 因 此 ， 如 果 按 照 列 empid 进行 分 组 并 执行 COUNT (sales) ， 就 会 按 组 显示 列 sales 的 值 的 
个 数 。 

另外 ， 因 为 计算 的 是 记录 的 数量 ， 所 以 在 没有 NULL 的 情况 下 ， 不 管 以 哪个 列 为 对 象 ， 结 果 都 
是 一 样 的 。 因 此 ,不 仅仅 是 COUNT (sales)， 执行 COUNT (empid) 或 COUNT (month) 也 不 会 有 
任何 问题 。 当 然 ， 执 行 COUNT (*) 也 很 方便 。 在 这 种 情况 下 ， 计 算 的 记录 数 就 包含 了 NULL。 

试 着 执行 下 面 的 命令 。 


SELECT COUNT (*) EROM tb GROUP BY empid; 


mysql> SELECT COUNT (*) FROM tb GROUP BY empid; 












































5 rows in set (0.00 sec) 


























执行 结果 中 显示 了 每 组 记录 的 个 数 ， 但 是 上 面 的 执行 结果 并 不 容易 让 人 明白 表示 的 是 什么 个 
数 ， 所 以 我 们 把 命令 修改 成 下 面 这 样 。 








SELECT 
empid, COUNT (*) RS 个 数 


FROM tb 
GROUP BY empid; 
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执行 结果 
mysql> SELECT empid,COUNT(*) AS 个 数 FROM tb GROUP BY empid; 
+------- +------ 十 

empiqd | 个 数 
+------- +------ 十 

Al01 2 

A1l02 2 

A1l03 

Al04 2 

Al107 是 
+------- +------ 十 
5 rows in set (0.00 sec) 























上 面 的 结果 中 显示 了 每 组 的 empid 和 COUNT (*) 的 值 。 为 了 便于 理解 ， 还 设置 了 “个 数 ” 这 
个 别名 。 











C 8.6.3 显示 各 组 的 总 和 以 及 平均 值 

下 面 我 们 试 着 计算 每 位 员工 的 总 销售 额 。 计 算 总 和 的 函数 是 SUM。 计 算 列 sales 的 总 和 时 需要 
使 用 SUM(sales) 。 

按照 列 empid 进行 分 组 ， 显 示 表 tb 中 每 组 销售 额 的 总 和 ， 并 加 上 “合计 ”这 个 别名 。 








PP 执行 内 容 


























































































































区 执行 前 ( 表 tb ) > 执行 后 
empid sales month empid | 合计 
A103 101 4 A101 484 
A102 54 5 A102 |259 
A104 181 4 A103 |130 
A101 184 4 A104 |274 
A103 17 5 A107 |87 
A101 300 5 9 9 
A102 205 6 显示 列 empid 
A104 93 5 和 每 组 的 总 和 
A103 12 6 
A107 87 6 

? 











按照 该 列 分 组 按 组 计算 列 sales 中 值 的 总 和 
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操作 方法 
执行 下 面 的 命令 。 
SELECT 
empid,SUM(sales) AS 合计 
FROM tb 


GROUP BY empid; 














mysql> SELECT empid,SUM(sales) AS 合计 FROM tb GROUP BY empid; 
ne 入 要 芝 训 芝 总 芝 十 
empid 合计 
下 人 + 
Al01 484 
A1l02 259 
A1l03 130 
A1l04 274 
A1l07 87 
Ve i + 
5 rows in set (0.00 sec) 











上 面 的 示例 显示 了 各 员工 的 总 销售 额 。 例 如 ,员工 号 为 “A101” 的 员工 ， 总 销售 人 额 为 184 + 
300， 即 484 万 元 。 
下 面 是 按 empid 分 组 并 计算 每 组 销售 额 平均 值 的 示例 。 


SELECT 
empid,AVG(sales) 


FROM tb 
GROUP BY empid; 











mysql> SELECT empid,AVG(sales) FROM tb GROUP BY empid; 
+------- 和 十 

| empid | AVG(sales) | 
+------- 站 说 + 

| Alol | 242.0000 | 

| A102 | 129.5000 | 

| A103 | 43.3333 | 

| A104 | 137.0000 | 

| A107 | 87.0000 | 
es i + 
5 rows in set (0.00 sec) 
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8.7 ”设置 条 件 分 组 显示 


8.7.1 


按 组 处 理 


一 一 人 一 一 一 一 





下 

















面 介绍 的 内 容 稍微 有 点 难 。 我 们 试 着 通过 GROUP BY 进行 分 组 ， 并 设置 记录 的 提取 条 件 。 








例如 ， 想 要 “ 按 员工 号 计算 销售 额 的 总 和 ， 但 是 仅 显 示 总 和 大 于 等 于 x x 的 记录 ”时 ， 可 以 





使 用 HAVING 为 分 组 的 结果 值 设置 提取 条 件 。 





分 组 并 设置 记录 的 提取 条 件 


用 HAVING 设置 的 条 件 适 






































] 于 分 组 的 结果 值 。 下 钙 


i 我 们 来 实际 操作 一 下 。 设 置 “ 按 员工 号 分 组 


计算 总 销售 额 ,， 但 仅 显 示 小 组 总 销售 额 大 于 等 于 200 万 元 的 记录 ”的 条 件 。 具 体 来 说 ， 就 是 对 于 表 
tb 中 的 每 一 种 empid， 显 示 “ 列 sales 的 总 和 大 于 等 于 200” 的 列 empid 及 列 sales 的 总 和 。 





PP 执行 内 容 























































































































> 执行 前 ( 表 tb ) > 执行 后 

empid | sales month empid | sales month empid SUM (sales) 

A103 101 4 A101 484 

A102 54 5 A102 259 

A104 181 4 A104 274 

A101 |184 |4 ?人 

A103 17 5 显示 列 empid 和 各 
组 总 销售 额 大 于 等 

A101 300 5 于 200 万 元 的 记录 

A102 205 6 

A104 93 5 

A103 1 肥 6 

A107 87 6 

和 本 
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操作 方法 




















SELECT 

empid,SsSUM(sales) 
FROM tb 

GROUP BY empid 
HAVING SUM(sales)>=200; 








mysql> SELECT empid,SUM(sales) FROM tb GROUP BY empid HAVING SUM(sales)>=200; 
+------- +------------ 十 
| empid | SUM(sales) | 


二 和 中 
| Alol | 484 | 
| Alo2 | 259 | 
| Alo4 | 274 | 
be A + 


3 rows in set (0.05 sec) 























如 果 没 有 写 最 后 的 HAVING SUM (sales)> = 200, 执行 结果 就 是 下 面 这 种 形式 。 


( 没有 记述 HAVING SUM (sales)>=200 ) 














mysql> SELECT empid,SUM(sales) FROM tb GROUP BY empid ; 
+------- +------------ 十 
empid SUM (sales) 
+------- +------------ 十 
Al01 484 
A1L02 汪汪 全 
A1L03 二 3 
Al04 274 
A1L07 87 
+------- +------------ 十 
5 rows in set (0.00 sec) 











从 上 面 的 执行 结果 中 可 以 看 到 ，A103 和 Al107 的 总 销售 额 都 小 于 200 万 元 ， 所 以 相关 记录 被 省 
略 了 。 

本 节 我 们 处 理 的 内 容 是 “ 按 员工 号 分 组 并 计算 各 组 总 销售 额 ， 但 仅 显 示 小 组 总 销售 额 大 于 等 于 
200 万 元 的 记录 ”。 也 就 是 说 ， 用 于 提取 记录 的 HAVING 是 在 分 组 之 后 执行 的 。 我 们 来 看 看 这 一 处 
理 方式 和 下 一 节 介绍 的 “提取 记录 后 分 组 ”有 什么 区 别 。 
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S 8.7.2 提取 记录 后 分 组 


与 前 一 节 的 “分 组 后 提取 记录 ”相反 ， 本 节 将 介绍 “提取 记录 后 分 组 ”的 相关 内 容 。 例 如 ， 仅 
提取 销售 额 大 于 等 于 1 万 元 的 交易 记录 ， 并 以 该 记录 为 对 象 计算 各 员工 的 平均 销售 额 。 

我 们 需要 使 用 WHERE (一 8.3.2 节 ) 执行 分 组 之 前 的 提取 操作 。 与 以 往 不 同 的 是 ， 使 用 GROUP 
BY 分 组 的 操作 要 放 在 最 后 面 。 

下 面 我 们 试 着 仅 提取 销售 额 (sales ) 大 于 等 于 50 万 元 的 交易 记录 ， 并 以 该 记录 为 对 象 计算 各 
员工 的 平均 销售 额 。 具 体 来 说 ， 就 是 对 于 表 tb 的 列 sales 中 大 于 等 于 50 的 记录 ， 按 照 列 empid 分 组 
后 ， 显 示 列 empid 以 及 各 组 sales 的 平均 值 。 


> 执行 内 容 


> 执行 前 ( 表 tb ) 




























































































empid sales month 
A103 101 4 
A102 54 5 
A104 181 4 
A101 ) 1 四 按 此 列 分 组 
A103 300 5 
AL 一 0 205 6 
A102 记录 93 5 
A104 87 6 
A103 
A107 

ww > 执行 后 
empid sales month 





empid | AvVG(sales) 
A101 242.0000 
A102 129.5000 
A103 101.0000 
A104 137.0000 
A107 87.0000 


7? 


显示 列 empid 和 各 组 
的 平均 值 

















将 














导 
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操作 方法 





执行 下 面 的 命令 。 




















SELECT 

empid,AVG (sales) 
FROM tb 

WHERE sales>=50 
GROUP BY empid; 














empid AVG (sales) 
+------- +------------ + 
AlL01 242.0000 
Al102 129.5000 
Al103 101.0000 
Al104 L370000 
Al107 87.0000 
+------- +------------ 十 


5 rows in set (0.02 sec) 











试 着 用 “SELECT empid,sales FROM tb;” 显 示 所 有 记录 并 确认 结果 。 





2E 汪 ( 显示 所 有 记录 ) 














mysql> SELECT empid,sales FROM tb; 
于 等 二 和 +------- 十 
empid sales 
+------- +------- 十 
Al103 101 
Al02 54 
Al04 181 
Al01 184 
Al03 1¥ 
Al01 300 
Al02 205 
Al04 93 
Al103 12 
AL07 87 
+------- +------- + 
10 rows in set (0.00 sec) 
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我 们 来 确认 一 下 提取 结果 。A101 在 列 sales 中 的 数据 有 2 个 ， 分 别 是 184 和 300。 这 两 个 数据 
都 满足 大 于 等 于 50 的 条 件 ， 所 以 它们 的 平均 值 为 242。 而 A103 在 列 sales 中 的 数据 有 3 个 ,分别 
是 101、17 和 12。 由 于 17 和 12 小 于 5$0， 所 以 被 省 略 掉 了 。 结 果 只 有 一 个 101 作为 平均 处 理 对 象 ， 
因此 平均 值 为 101。 


8.7.3 分 组 后 排序 


我 们 试 着 对 分 组 的 结果 重新 排序 并 将 结果 显示 出 来 。 也 就 是 需要 同时 使 用 “GROUP BY ...” 
和 “ORDER BY ...”。 记 述 时 先 写 “GROUP BY ..”， 再 写 “ORDER BY ...”。 

试 着 将 表 tb 各 员工 的 平均 销售 额 (AVG (sales) ) 按 从 高 到 低 的 顺序 显示 。 具 体 来 说 ， 就 是 
对 表 tb 按照 列 empid 进行 分 组 ,然后 按照 各 组 平均 销售 额 从 高 到 低 的 顺序 显示 列 empid 和 各 组 销售 
额 的 平均 值 。 





> 执行 内 容 




































































> 执行 前 ( 表 tb ) > 执行 后 
sales month empid | AvVG(sales) 
101 4 A101 
54 a A104 
181 4 A102 
184 4 A107 
1 5 A103 
| 
205 6 
93 5 
12 6 
87 6 




















3 
按 此 列 分 组 计算 各 组 平均 值 
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操作 方法 





执行 下 面 的 命令 。 

















， SELECT 

empid,AVG (sales) 
: FROM tb 

: GROUP BY empid 

， ORDER BY AVG (sales) 














DESC; 

执行 结果 

mysql> SELECT empid,AVG(sales) FROM tb GROUP BY empid ORDER BY AVG(sales) DESC; 

+------- 人 + 
empid AVG (sales) 

和 Ee + 
Al01 242.0000 
Al04 137;0000 
Al102 129.5000 
Al107 87.0000 
A1l03 43.3333 

+------- +------------ 十 


5 rows in set (0.00 sec) 











可 以 看 到 ， 显 示 的 内 容 按照 empid 分 组 ， 并 按照 各 组 销售 额 的 平均 值 降序 排列 。 





S 8.7.4 ”分 组 方法 总 结 


下 面 总 结 一 下 前 面 提 到 的 在 分 组 的 情况 下 设置 条 件 的 方法 ， 主 要 包括 以 下 两 种 类 型 。 




















a. 提取 记录 后 分 组 
b. 分 组 后 提取 记录 


在 a. 的 情况 下 ， 需 要 使 用 WHERE 设置 条 件 并 提取 记录 ， 然 后 通过 GROUP BY 进行 分 组 。 在 
b. 的 情况 下 ， 需 要 先 用 GROUP BY 进行 分 组 ， 然 后 使 用 HAVING 提取 记录 。 此 外 ， 我 们 使 用 了 
ORDER BY 对 分 组 后 的 结果 进行 排序 。 

我 们 试 着 把 这 些 方法 组 合 起 来 进行 处 理 。 以 销售 额 大 于 等 于 50 万 元 的 数据 为 对 象 ， 按 照 员工 
号 (empid ) 分 组 ,计算 各 员工 的 平均 销售 额 ( AVG (sales) )， 然 后 按照 降序 显示 。 
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> 执行 内 容 
> 执行 前 ( 表 tb ) 





Sales month 
101 
54 

181 

















按 此 列 分 组 











184 
300 
205 
93 
87 

















提取 列 值 大 于 
等 于 50 的 记录 























olalolaleells|ialn 





























降序 显示 

















额 平均 值 


























操作 方法 


执行 下 面 的 命令 。 





SELECT 

empid,AVG (sales) 
FROM tb 

WHERE sales>=50 
GROUP BY empid 

ORDER BY AVG (sales) 
DESC; 


mysql> SELECT empid,AVG(sales) FROM tb WHERE sales>=50 GROUP BY empid ORDER BY 
AVG (salLes) DESC; 














+------- i + 
empid | AVG(sales) 
+------- +------------ 十 
Al01 242.0000 
Al04 L370000 
A1l02 129.5000 
A1l03 101.0000 
A1l07 87.0000 
+------- 二 + 








5 rows in set (0.00 sec) 





当 涉 及 提取 、 排 序 和 分 组 时 ， 就 连 MySQL 专家 也 需要 考虑 怎么 写 SQL 语句 才 好 。 在 使 用 
SELECT、GROUP BY 和 ORDER BY 等 多 个 语句 的 情况 下 ， 最 重要 的 就 是 记述 顺序 。 如 果 弄 错 了 顺 
序 ， 程 序 就 会 发 生 错误 ， 这 一 点 需要 大 家 注意 。 

WHERE、GROUP BY 和 ORDER BY 的 记述 顺序 如 下 所 示 。 





G@) WHERE 条 件 
vv 
@ GROUP BY ... 
vv 
(ORDER BY ... (DESC ) 


本 章 介绍 了 以 下 内 容 。 


@ 设 置 条 件 提取 记录 的 方法 

@ 使 用 函数 提取 记录 的 方法 

@ 按 照 指 定 条 件 排序 并 显示 的 方法 
@ 指 定 显示 的 记录 数 提取 记录 的 方法 
@ 设 置 多 个 条 件 提 取 记 录 的 方法 

@ 分 组 显示 的 方法 























当 WHERE、GROUP BY 和 ORDER BY 纠缠 在 一 起 时 ,语句 就 会 变 得 很 复杂 。 如 果 能 够 准 而 
握 记述 的 顺序 ， 工 作 效 率 就 会 得 到 大 幅 提 升 。 





迟 
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自我 检查 
下 面 检查 一 下 本 章 学 习 的 内 容 是 否 全 部 理解 并 掌握 了 。 




















口 能 够 通过 WHERE 设置 条 dee SELECT 

口 能 够 通过 LIMIT 限制 记录 的 数量 进行 SELECT 

口 能 够 使 用 “ORDER BY ... ( DESC )” 按 照 升序 ( 降序 ) 进行 SELECT 
口 能 够 使 用 GROUP BY 分 组 SELECT 

口 能 够 通过 WHERE 和 HAVING 设置 条 件 ， 使 用 GROUP BY 进行 SELECT 
口 能 够 理解 WHERE、GROUP BY 和 ORDER BY 的 记述 顺序 





二 练习 题 





问题 1 


请 在 表 tb 的 列 sales 的 总 和 前 后 加 上 字符 串 “ 合 计 ” 和 “万 元 ”并 显示 出 来 。 另 外 ， 在 显示 的 项 目 名 
上 添加 别名 “销售 额 "。 






































字符 串 的 组 合 、SUM 函数 和 别名 等 内 容 在 正文 中 都 有 介绍 。 请 大 家 认真 思考 。 















































问题 2 


在 下 面 的 表 tb 中， 以 列 sales 大 于 等 于 50 的 数据 为 对 象 ， 按 照 列 empid 分 组 ， 并 降序 显示 各 组 销售 




















叫 





































































































额 平均 值 大 于 等 于 120 的 记录 。 
PF 表 tb 
empid sales month 
A103 101 4 
A102 54 5 
A104 181 4 
A101 184 4 
A103 以 局 
A101 300 加 
A102 205 6 
A104 98 局 
A103 1 要 6 
A107 87 6 

2 该 习题 与 正文 8.7.1 节 的 示例 相同 。 只 要 加 上 HAVING 即 可 。 











it 参考 答案 














问题 1 
执行 下 面 的 命令 。 


SELECT CONCAT(' 合 SUM(sales),' 万 元 ') AS 销售 额 FROM tb; 





mysql> SELECT CONCAT(' 合 计 ',SUM(sales),' 万 元 ') RS 销售 额 FROM tb; 





| 销售 额 | 
i 十 
| 合计 1234 万 元 | 
A 十 


1 row in set (0.00 sec) 

















问题 2 
执行 下 面 的 命令 。 


SELECT empid,AVG(sales) 
FROM tb 
WHERE sales>=50 


GROUP BY empid 
HAVING AVG(sales) >=120 
ORDER BY AVG(sales) DESC; 





mysql> SELECT empid,AVG(sales) FROM tb WHERE sales>=50 GROUP BY empid HAVING 
AVG (sales)>=120 ORDER BY AVG(sales) DESC; 





+------- a + 
| Alol | 242.0000 | 
| Al04 | 137.0000 | 
| Al02 | 129.5000 | 
下 下 + 


3 rows in set (0.00 sec) 
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第 8 章 使 用 各 种 条 件 进 行 提取 


WHERE 和 HAVING 



































E 文 中 介绍 了 WHERE 和 HAVING 在 途 





1 


ORDER BY 和 HAVING 的 SELECT 语句 通 


Ss 


但 是 ， 实 际 的 执行 顺序 却 是 下 面 这 样 的 。 




















方法 上 的 区 别 ， 这 里 我 们 再 来 总 结 一 下 。 例 如 ， 信 




















ELECT ~ FROM ~ WHERE ~ GRO 


























常会 按照 如 下 方式 描述 。“ ~ ”部 分 是 可 选 的 。 








UP BY ~ HAVING ~ ORDER BY ~ 


EROMY > WHERE ”EROUVP BY HAVING >SEIECTI ORDEREY 


也 就 是 说 ， 在 通过 GROUP BY 分 组 之 前 会 先 执行 WHERE， 而 HAVING 执行 的 对 象 是 GROUP BY 分 组 








不 o 


外 ， 可 以 看 到 ORDER BY 重新 排列 了 





SELECT 的 结果 。 











GROUBEEYS 





ol 


口 | 


第 9 章 编 


9.1 更 新 记录 


辑 数据 























掌握 提取 记录 的 方法 之 后 ， 我 们 来 编辑 一 下 数据 。 数 据 的 编 














管理 员 最 为 敏感 的 部 分 。 


辑 包 括 修改 各 个 列 的 数据 、 删 除 或 复制 记录 等 操作 。 这 是 数据 库 








@y9.11 瞬间 更 新 列 中 所 有 的 记录 


首先 是 修改 记录 的 方法 。 修 改 记录 时 需要 使 用 UPDATE 命令 


修改 列 的 所 有 记录 


: UPDATE 表 名 SET 列 名 = 设 


入 置 的 值 ; 



































上 面 的 命令 会 给 指定 的 列 设置 值 。 如 果 执 行 这 个 命令 ， 不 管 有 几 千 条 还 是 几 万 条 记录 ， 列 中 的 





所 有 记录 都 会 被 瞬间 替换 掉 。 


UPDATE 命令 通常 在 通过 WHERE 设置 条 





件 之 后 ， 以 特定 的 记录 为 对 象 执 行 。 如 果 没 有 使 











LU 





WHERE 设置 条 件 ， 所 有 的 列 都 会 被 替换 掉 ， 因 此 必须 小 心 处 理 。 








下 面 我 们 来 实际 操作 一 下 。 








(一 6.3 方 )， 然 后 把 数据 插入 列 











| 


o 


列 值 一 旦 被 修改 就 很 难 再 复原 ， 所 以 我 们 来 创建 一 个 新 的 列 





现在 销售 信息 表 tb 中 只 有 员工 号 empid、 销 售 额 sales 和 月 份 month 这 3 个 列 ， 我 们 再 添加 一 





个 数据 类 型 为 VARCHAR (100) 


的 列 remark 来 表示 “备注 ” 





。 使 用 UPDATE 命令 将 列 remark 的 所 有 
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记录 更 新 为 “无 特殊 记录 ”， 然 后 显示 所 有 记录 。 





> 执行 内 容 





































































































> 执行 前 ( 表 tb ) > 执行 后 

empid | sales month empid Sales month remark @ -添加 列 
A103 |101 4 A103 101 4 无 特殊 记录 和 
A102 |54 5 A102 54 5 无 特殊 记录 输 入 “无 
A104 |181 4 A104 |181 4 无 特殊 记录 特殊 记录 
A101 |184 4 A101 184 4 无 特殊 记录 

A103 |17 5 少 A103 17 5 无 特殊 记录 

A101 |300 5 A101 300 5 无 特殊 记录 

A102 |205 6 A102 205 6 无 特殊 记录 

A104 |93 5 A104 93 5 无 特殊 记录 

AT103 | 人 2 6 A103 届 6 无 特殊 记录 

A107 |87 6 A107 87 6 无 特殊 记录 

操作 方法 


中 执行 下 面 的 命令 。 









































mysql> ALTER TABLE tb ADD remark VARCHAR(100); 
Query OK, 0 rows affected (0.17 sec) 


Records: 0 Duplicates: 0 Warnings: 0 
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mysql> UPDATE tb SET remark=' 无 特殊 记录 '; 
Query OK, 10 rows affected (0.00 sec) 
Rows matched: 10 Changed: 10 Warnings: 0 





mysql> SELECT * FROM tb; 


























+------- +------- +------- 下 十 
empid sales month remark 

+------- +------- +------- +------------ 十 
ARA103 101 4 无 特殊 记录 
ARA102 54 5 无 特殊 记录 
ARA104 181 4 无 特殊 记录 
A101 184 4 无 特殊 记录 
RAR103 17 5 无 特殊 记录 
Al01 300 5 无 特殊 记录 
A1l02 205 6 无 特殊 记录 
Al04 93 5 无 特殊 记录 
ARA103 1 6 无 特殊 记录 
A107 87 6 无 特殊 记录 

+------- +------- +------- +------------ 十 


10 rows in set (0.00 sec) 




















执行 UPDATE 命令 后 ， 结 果 中 会 显示 x xrows affected， 它 表示 x x 条 记录 受到 影响 。 
在 本 例 中 ， 因 为 没有 设置 WHERE 条 件 ， 所 以 列 remark 的 所 有 记录 了 瞬间 变 成 了 “无 特殊 记录 ”。 





防止 意外 执行 UPDATE 和 DELETE 
当 我 们 在 工作 中 使 用 MySQL 时 ， 虽 然 有 时 需要 一 次 性 更 新 列 的 所 有 内 容 ， 但 这 只 是 偶尔 的 事情 。 如 果 不 慎 
将 所 有 的 列 都 更 新 成 相同 的 值 就 麻烦 了 。 
所 以 ， 为 了 防止 这 种 情况 发 生 ， 在 启动 MySQL 监视 器 的 时 候 可 以 加 上 --safe-updates 选项 。 在 使 用 此 
选项 的 情况 下 ， 如 果 列 上 没有 WHERE 条 件 就 无 法 执行 UPDATE 或 DELETE。 

这 是 一 个 适合 数据 库 管 理 初学 者 使 用 的 功能 。 





























































































































修改 列 中 所 有 的 记录 虽然 在 操作 上 非常 简单 ， 但 实际 上 执行 的 机 会 并 不 多 。 很 多 时 候 ， 我 们 只 
需要 修改 符合 条 件 的 记录 。 
我 们 试 着 只 修改 符合 WHERE 条 件 的 记录 ， 有 具体 命令 如 下 所 示 。 


只 修改 符合 条 件 的 记录 
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上 面 的 





命令 只 是 


节 )、LIMIT (一 8.3.1 节 ) 来 设置 条 件 。 

在 9.1.1 节 中 ， 列 remark 中 全 部 输入 了 “无 特殊 记录 ”。 查 看 销售 信息 表 tb 就 会 发 现 有 几 个 销 
售 额 超过 100 万 元 的 优秀 成 绩 。 因 此 ， 我 们 以 列 sales 大 于 等 于 100 的 记录 为 对 象 ， 向 备注 ( 列 
remark ) 中 输入 “优秀 ”。 


P 执行 内 容 
> 执行 前 ( 表 tb ) 

















在 UPDATE 中 添加 了 WHERE 条 件 。 当 然 ， 我 们 也 可 以 使 用 ORDER BY (一 8.5 












































































































































empid | sales month | remark 
A103 101 4 无 特殊 记录 
A102 54 5 无 特殊 记录 
A104 181 4 无 特殊 记录 
A101 184 4 无 特殊 记录 
A103 17 5 无 特殊 记录 
A101 300 5 无 特殊 记录 
A102 |205 6 无 特殊 记录 
A104 |93 5 无 特殊 记录 
A103 12 6 无 特殊 记录 
A107 |87 6 无 特殊 记录 
9 
列 值 大 于 等 于 100 的 记录 
操作 方法 


中 执行 下 面 的 命令 。 














区 执行 后 

empid | sales month | remark 
A103 ”Ml 4 优秀 

A102 |54 5 无 特殊 记录 
A104 |181 4 优秀 

A101 1184 4 优秀 

A103 117 5 无 特殊 记录 
A101 | 300 5 优秀 

A102 |205 6 优秀 

A104 |93 5 无 特殊 记录 
A103 |12 6 无 特殊 记录 
A107 |87 6 无 特殊 记录 














输入 "优秀 ” 
































mysql> SELECT * FROM tb; 


+------- +------- +------- +------------ + 


| empid | sales | month | remark 
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+------- +------- +------- +------------ 十 
ARA103 101 4 万 秀 
RA102 54 5 | 无 特殊 记录 
ARA104 181 4 万 秀 
Al01 184 4 万 秀 
A1l03 lg 5 | 无 特殊 记录 
Al01 300 5 万 秀 
A1l02 205 6 侈 秀 
RA104 93 5 | 无 特殊 记录 
ARA103 12 6 | 无 特殊 记录 
A107 87 6 | 无 特殊 记录 
+------- +------- +------- +------------ 十 
10 rows in set (0.00 sec) 

















可 以 看 到 ,没有 更 新 为 “优秀 ”的 记录 ， 其 内 容 还 是 “无 特殊 记录 ”。 


S 9.1.3 ”将 销售 额 最 低 的 3 条 记录 的 备注 修改 为 “加 油 !” 

在 上 一 节 中 ,我们 以 销售 额 大 于 等 于 100 万 元 的 记录 为 对 象 ， 在 符合 条 件 的 备注 中 输入 了 “ 优 
秀 "”。 这 次 我 们 试 着 给 销售 额 较 低 的 记录 输入 鼓励 的 信息 。 给 所 有 销售 额 排序 ， 并 将 “加 油 !” 输 入 
到 销售 额 最 低 的 3 条 记录 的 备注 中 。 那 么 , “销售 额 最 低 的 3 条 记录 ”这 一 条 件 该 如 何 设置 呢 ? 
虽然 看 起 来 有 点 麻烦 ,但 总 的 来 说 是 使 用 ORDER BY 将 列 sales 按 升序 排列 ， 并 用 LIMIT 3 
对 前 3 条 记录 进行 SELECT， 然 后 向 列 remark 中 输入 “加 油 !”， 最 后 执行 UPDATE。 


PP 执行 内 容 























































































































> 执行 前 ( 表 tb ) > 执行 后 

empid | sales month | remark empid | sales month | remark 
A103 |101 4 人 万 秀 A103 |101 4 优秀 

A102 64 5 无 特殊 记录 A102 li54 5 加 油 ! 
A104 181 4 优秀 A104 |181 4 优秀 

A101 184 4 优秀 A101 184 4 优秀 

A103 “ 国 加 5 无 特殊 记录 >》 A103 Bm 6 加 油 ! 

A10 300 5 优秀 A101 |300 5 优秀 

A102 |205 6 优秀 AT102 1205 6 优秀 

A104 93 5 无 特殊 记录 A104 |93 5 无 特殊 记录 
A103 “ 国 思 6 无 特殊 记录 A103 112 6 加 油 ! 
A107 |87 6 无 特殊 记录 A107 |87 6 无 特殊 记录 










































































最 低 的 3 条 记录 输入 加油!” 
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操作 方法 


db 执行 下 面 的 命令 。 


UPDATE tb 

SET remark=' 加 油 ! ' 
ORDER BY sales 

LIMIT 3; 





mysql> UPDATE tb SET remark=' 加 油 !' ORDER BY sales LIMIT 3; 
Query OK, 0 rows affected (0.04 sec) 
Rows matched: 3 Changed: 0 Warnings: 0 


mysql> SELECT * FROM tb; 

































































+------- +------- +------- et + 

empid sales month remark 
+------- +------- +------- 津 皇 庆 :S 二 二 汪 二 二 二 二 :二 二 + 

A103 101 4 优秀 

RA102 54 5 加 油 ! 

A104 181 4 优秀 

A101 184 4 优秀 

A103 7 5 | 加 油 ! 

A101 300 5 优秀 

A102 205 6 优秀 

A104 93 5 | 无 特殊 记录 

RA103 12 6 | 加 油 ! 

A107 87 6 | 无 特殊 记录 
+------- +------- +------- 沸 宇 SSS + 


10 rows in set (0.00 sec) 











是 否 真 的 只 更 新 了 销售 额 最 低 的 3 条 记录 呢 ? 我 们 试 着 执行 下 面 的 命令 ， 按 销售 额 由 低 到 高 的 
顺序 排序 来 确认 一 下 结果 。 


SELECT * FROM tb ORDER BY sales; 
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mysql> SELECT * FROM tb ORDER BY sales; 
汕 吉 二 于 二 宇和 囊 生 和 和 十 

empid sales month remark 
4 中 二 = 4 十 

ARA103 了 和 6 | 加 ; 

ARA103 17 5 | 加 油 ! 

A102 54 5 加 油 ! 

A107 87 6 | 无 特殊 记录 

ARA104 93 5 | 无 特殊 记录 

ARA103 101 4 | 优秀 

ARA104 181 4 | 优秀 

Al01 184 4 | 优秀 

A1l02 205 6 | 优秀 

Al01 300 5 | 优秀 
再 二 SR 二 SS 4 十 
10 rows in set (0.02 sec) 











可 以 看 到 ， 只 有 销售 额 最 低 的 3 条 记录 更 新 了 。 

如 果 使 用 OFFSET (一 8.5.3 节 )， 还 可 以 对 从 第 x x 条 到 第 x x 条 的 记录 进行 更 新 。 

为 了 方便 今后 的 学 习 ， 我 们 要 把 销售 信息 表 tb 的 列 remark 删 掉 ， 然 后 恢复 成 原来 的 表 tb 
(一 8.1 节 )。 请 使 用 以 下 命令 (一 6.6 节 ) 删除 列 remark。 


ALTER TABLE tb DROP remark; 




















9.2 复制 符合 条 件 的 记录 


S 9.2.1 仅 复制 指定 记录 

第 7 章 介绍 了 3 种 复制 所 有 记录 的 方法 。 这 里 我 们 将 学 习 把 符合 条 件 的 记录 复制 到 其 他 表 中 的 方 
法 。 比 如 只 收集 成 绩 大 于 等 于 80 分 的 记录 或 者 只 收集 渡 边 的 记录 ， 然 后 重新 创建 一 个 表 这 样 的 操作 。 

使 用 CREATE TABLE 创建 一 个 新 表 ， 并 插 人 按照 条 件 提取 的 记录 。 也 就 是 说 ， 在 “CRERATE 
TABLE 新 表 名 SELECT * FROM 元 表 名 ”的 基础 上 ， 使 用 wHERE 设置 条 件 。 

试 着 从 销售 信息 表 tb 中 提取 资深 员工 “A101” 的 记录 ， 创 建 员 工 “A101” 的 专用 表 tb_A101。 
具体 来 说 ， 就 是 复制 表 tb 的 列 结构 和 empid 为 A101 的 记录 ， 然 后 创建 新 表 tb_A101。 创建 完成 后 ， 
试 着 显示 表 tb_A101 的 所 有 记录 。 
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> 执行 内 容 





































































































> 执行 前 ( 表 tb ) > 执行 后 ( 表 tb_A101 ) 
empid sales month empid sales month 
A103 101 4 村 A101 184 4 @ 一 创建 新 表 
A102 54 5 A101 300 加 
A104 181 4 
A101 184 4 

只 复制 A101 A103 17 5 

的 记录 
A101 300 
A102 205 6 
A104 93 5 
A103 12 6 
A107 87 6 

操作 方法 


中 执行 下 面 的 命令 。 


CREATE TABLE tb A1L01 
SELECT * 
FROM tb 
WHERE empid LIKE 'A101'; 





























mysql> CREATE TABLE tb A101 SELECT * FROM tb WHERE empid LIKE 'A1l01'; 
Query OK, 2 rows affected (0.03 sec) 
Records: 2 Duplicates: 0 Warnings: 0 


mysql> SELECT * FROM tb A1l01; 


尝 二 于 过 后 扎 吕 全 品名 路 二 中 
| empid | sales | month | 
hn > ht 中 
| Alol | 184 | 4 | 
| A1l01 | 300 | s | 
Fes FS Es 十 





2 rows in set (0.00 sec) 
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如 果 要 将 记录 插入 到 已 存在 的 表 中 ， 只 要 像 下 面 这 样 将 CREATE TABLE 的 部 分 改 为 INSERT 
INTO 即 可 。 


INSERT INTO 已 存在 的 表 SELECT * FROM tb WHERE empid LIKE 'A1l01'; 


S 9.2.2 ”排序 后 复制 
下 面 将 学 习 “ 只 复制 值 最 小 的 3 条 记录 “复制 从 第 5 条 到 第 10 条 的 记录 ”这 种 按照 列 值 的 顺 
序 复制 记录 的 方法 。 与 上 一 节 一 样 ， 当 执行 “CREATE TABLE ... SELECT ...” 时 , 使 用 ORDER 
BY 排序 ， 然 后 使 用 LIMIT 和 OFFSET 指定 要 复制 的 记录 数 和 开始 复制 记录 的 位 置 。 
下 面试 着 提取 销售 信息 表 tb 中 ， 按 照 销 售 额 从 高 到 低 的 顺序 排 在 第 2 名 到 第 5 名 的 记录 。 请 
参考 9.1.3 节 中 将 销售 额 最 低 的 3 条 记录 的 备注 修改 为 “加 油 !” 的 内 容 。 
具体 来 说 ， 就 是 复制 表 tb 的 列 结 构 以 及 按照 列 sales 的 值 由 高 到 低 的 顺序 复制 排 在 第 2 名 到 第 
5 名 的 4 条 记录 ， 创 建新 表 tb_2to5。 



































































































































> 表 的 结构 
> 执行 前 ( 表 tb ) 上 执行 后 ( 表 tb_2to5 ) 
empid sales month empid sales month 
第 5 名 JAT03 ot |4 A102 |205 |6 
A102 54 5 》 A101 184 4 
芋 A104 181 4 A104 181 4 
第 3 名 A Pea A103 | 101 4 
A103 17 5 
第 1 名 [A107 S00 |5 
第 2 名 AT07 lo |6 
A104 93 5 
A103 12 6 
A107 87 6 
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操作 方法 
执行 下 面 的 命令 。 


CREATE TABLE tb 2to5 
SELECT * 
FROM tb 
ORDER BY sales 
DESC 
LIMIT 4 
OFFSET 1; 





mysql> CREATE TABLE tb 2to5 SELECT * FROM tb ORDER BY sales DESC LIMIT 4 
OFFSET 1; 
Query OK, 4 rows affected (0.03 sec) 


Records: 4 Duplicates: 0 Warnings: 0 


mysql> SELECT * FROM tb 2to5; 


es Ess SS 十 
| empid | sales | month | 
和 ss es 中 
| A102 | 205 | 6 | 
| A1l01 | 184 | 4 | 
| Alo4 | 181 | 4 | 
| A103 | 101 | 4 | 
i i qe + 


4 rows in set (0.00 sec) 











这 样 ， 满 足 上 述 条 件 的 表 tb_2to5 就 创建 成 功 了 。 


9.3 ”删除 符合 条 件 的 记录 


931 删除 所 有 记录 ( 复 3) 


在 9.2.1 节 中 ， 我 们 复制 了 指定 记录 。 这 次 我 们 来 删除 指定 记录 。 因 为 记录 会 消失 ， 所 以 在 执 
行 相关 命令 的 时 候 要 上 比 执行 UPDATE 更 加 谨慎 才 行 。 
删除 所 有 记录 的 方法 是 执行 “DELETE FROM 表 名 ;” 命 令 (一 7.9 节 )。 























DELETE 








7 市 )。 














下 面 我 们 来 学 习 删 除 符合 条 件 的 记录 的 方法 。 前 面 已 
在 只 要 将 SELETE 替换 为 DELETE 即 可 。 使 月 








删除 符合 条 件 的 记录 


: DELETE FROM 表 名 WHERE 条 件 ; 


经 介绍 
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命令 虽然 会 删除 记录 ,但 它 并 不 会 删除 表 的 列 结 构 。 删 除 表 本 届时 需要 使 用 DROP 


TABLE 命令 (一 


了 通过 WHERE 条 件 提取 记录 的 方法 ， 现 
月 DELETE FROM 删除 记录 时 需要 通过 WHERE 来 设置 条 件 。 


通过 查看 员工 信息 表 t 了 bl 可 以 得 知 ， 员 工 年 龄 在 23~40 岁 。 我 们 来 删除 年 龄 小 于 30 岁 的 员工 的 记录 。 


在 做 删除 练习 时 ， 大 家 可 以 参考 第 7 章 的 内 容 ， 准 备 好 合适 的 表 之 后 再 进行 操作 。 这 里 我 们 使 





用 与 员工 信息 表 tbl 内 容 相同 的 表 tb1I。 
删除 表 tb1I 中 列 age 的 值 小 于 30 的 记录 。 删 除 记 录 后 ， 试 着 显示 表 tb1I 的 所 有 记录 。 


PP 执行 内 容 






































































































































> 执行 前 ( 表 tb11) > 执行 后 

empid name age empid name age 
A101 佐 茧 40 >》 A101 大 项 40 
A102 高 桥 28 A105 泽 35 
A103 中 川 20 

A104 渡 边 2 

A105 泽 35 

删除 列 值 小 于 30 的 记录 
操作 方法 
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mysql> DELETE FROM tblI WHERE age<30 
Query OK, 3 rows affectedq (0.07 sec) 


mysql> SELECT * FROM tblI: 


























+------- +------ +------ 十 
| empid | name | age | 
+------- +------ +------ 十 
| A101 | 会 芯 | 40 | 
| A105 | 西 泽 | 35 | 
+------- +------ +------ 十 
2 rows in set (0.00 sec) 




















列 age 中 值 小 于 30 的 记录 被 删除 了 ， 大 于 等 于 30 的 记录 被 保留 了 下 来 。 


本 节 是 9.2.2 节 “ 排 序 后 复制 ”的 删除 版 本 。 举 例 来 说 ， 就 是 删除 排 在 最 前 面 的 3 条 记录 这 种 情况 。 
对 于 此 类 处 理 ， 我 们 需要 使 用 ORDER BY 进行 排序 ， 然 后 使 用 LIMIT 指定 要 删除 的 记录 数 (一 8.3.1 节 ) 

针对 和 销售 信息 表 tb (一 8.1.1 节 ) 内 容 相同 的 表 tb_copy， 我 们 试 着 删除 销售 额 排 在 最 前 面 的 
4 条 记录 。 具 体 来 说 ， 就 是 删除 列 sales 中 值 最 大 的 4 条 记录 ， 然 后 显示 表 tb_copy 的 所 有 记录 。 

另外 ， 我 们 可 以 使 用 “CRERATE TABLE tb copy SELECT * FROM tb;” 命 令 (一 7.2 节 ) 
创建 与 表 也 内 容 相同 的 表 tb_copy。 


PP 执行 内 容 

































































































































































> 执行 前 ( 表 tb_copy ) > 执行 后 
empid | sales month empid | sales month 
A103 101 4 A103 101 4 
A102 54 5 A102 54 5 
第 4 名 上 el A104 ”|181 4 本 A103 117 5 
第 3 名 一 一 ®@)Al0l 184 4 A104 93 5 
A103 17 5 A103 12 6 
第 1 名 Fe A101 300 区 A107 87 6 
第 2 名 一 A102 吕 05 6 
A104 93 5 
A103 12 6 
A107 87 6 
删除 该 列 中 值 最 大 的 4 条 记录 

















操作 方法 





中 执行 下 面 的 命令 。 

















DELETE FROM tb copy 
ORDER BY sales 
DESC 
LIMIT 4; 























mysql> DELETE FROM tb copy ORDER BY sales DESC LIMIT 4; 
Query OK, 4 rows affected (0.00 sec) 





mysql> SELECT * FROM tb copy: 














+------- +------- +------- 十 
empid sales month 
+------- +------- +------- 十 
Al103 461 4 
Al102 54 5 
Al103 3 S 
Al104 23 S 
A1l03 2 6 
A1l07 87 6 
+------- +------- +------- 十 


6 rows in set (0.00 sec) 











和 原来 的 表 相 比 ， 可 以 看 到 列 sales 中 值 最 大 的 4 条 记录 被 删除 了 。 


9.4 总 结 


本 章 介绍 了 以 下 内 容 。 


@ 如 何 更 新 指定 列 的 所 有 值 
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@ 如 何 更 新 符合 条 件 的 记录 
@ 如 何 删除 符合 条 件 的 记录 
@ 如 何在 按照 升序 ( 降序 ) 排列 的 记录 中 指定 记录 条 数 更 新 列 值 


自我 检查 
下 面 检查 一 下 本 章 学 习 的 内 容 是 否 全 部 理解 并 掌握 了 。 




















口 能 够 使 用 UPDATE 命令 更 新 列 值 

口 能 够 使 用 WHERE 更 新 符合 条 件 的 记录 

口 能 够 使 用 WHERE 复制 符合 条 件 的 记录 

口 能 够 使 用 ORDER BY 和 LIMIT 对 指定 记录 进行 更 新 
口 能 够 使 用 WHERE 删除 符合 条 件 的 记录 


































































































练习 题 
问题 1 
表 StOCK 下 用 示 @ 
表 t_stock 
列 名 a b C 
列 的 数据 类 型 VARCHAR (10) INT DATE 
于 2 200 2011-08-08 
分 店 500 2017-06-15 
中 的 突 
”> 分店 100 2010-02-23 
了 分 店 400 2019-08-08 
提取 表 t_stock 的 列 c 中 除 距 今 5 年 前 以 外 的 值 的 记录 并 创建 表 t_stock_new。 这 个 “5 年 前 ”的 时 间 
下 面 的 命令 表示 。 假 设 这 个 处 理 是 在 2018 年 进行 的 。 


















































国 5 年 前 的 时 间 





























T 
人 | 请 记 住 INTERVAL x x YEAR 这 个 便利 的 用 法 。 








ta 参考 答案 














问题 1 
执行 下 面 的 命令 。 


CREATE TABLE t stock new 


SELECT * FROM t stock 
where C >NOW() - INTERVAL 5 YEAR; 





我 们 来 确认 一 下 表 t_stock 的 内 容 。 
































mysql> SELECT * FROM t stock; 
刷 二 全 全 全 +------ Eee 十 
| a | b | < | 
es 机 下 中 十 
| 东 分 店 | 200 | 2011-08-08 | 
| 西 分 店 | 500 | 2017-06-15 | 
| 南 分 店 | 100 | 2010-02-23 | 
| 北 分 店 | 400 | 2019-08-08 | 
本 二 ee 十 
4 rows in set (0.00 sec) 











创建 并 确认 表 t_stock_new 的 内 容 。 


mysql> CREATE TABLE t stock new SELECT * FROM t_stock WHERE c> NOW() - INTERVAL 
5 YEAR; 
Query OK, 2 rows affected (0.27 sec) 





Records: 2 Duplicates: 0 Warnings: 0 


mysql> SELECT * FROM t stock new; 























第 字 字 和 Eo 让 是 十 
| a | b | < | 
Fs 乓 SS Fo 十 
| 西 分 店 | 500 | 2017-06-15 | 
| 北 分 店 | 400 | 2019-08-08 | 
+-------- +------ 吕 二 + 


2 rows in set (0.00 sec) 
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数据 库 的 运用 方法 

















本 书 是 针对 使 
者 编写 的 。 但 在 实际 












































@ 本 地 部 署 


本 地 部 署 ( on-premise ) 








由 廊 法 8 






































意 强调 本 地 部 署 这 个 概念 。 




















于 需 

















专业 知识 。 特 别 是 近 几 年 ， 




















系统 安全 方 

















的 问题 也 必须 








方法 和 





各 种 各 样 的 形式 。 虽 然 不 能 明确 地 i 


是 个 人 或 组 织 拥有 设备 来 使 用 系统 的 一 种 


自己 完成 所 有 工作 ， 所 以 初期 成 本 


AMP 在 PC 上 安装 MySQL、Apache 和 PHP， 并 在 localhost 环境 下 学 习 
作 中 ，Web 服务 器 和 数据 库 的 运 
但 我 们 可 以 看 一 下 以 下 3 种 运 























形式 。 过 去 只 有 这 一 种 形式 


很 高 ， 操 作 和 维护 也 需要 用 

















顾 。 但 

















， 本 地 部 署 在 设计 的 修改 




















统合 作 等 方面 有 较 强 的 灵活 性 。 








信和 租赁 服务 器 


租赁 服务 器 ( rental server ) 是 供应 商 租 赁 系统 供用 户 使 


务 器 。 服 务 器 的 管理 和 维护 由 供应 商 负责 ， 所 以 不 需要 用 到 高 级 的 专业 知识 ， 初 期 成 本 也 能 大 大 缩减 。 在 大 多 数 










































































情况 下 ，MySQL 等 数据 库 环境 只 会 使 用 供 


ez 


云 ( cloud ) 是 租借 云 环 


的 系统 环境 中 的 CPU 和 存 














应 商事 前 准备 的 产品 。 


的 一 种 形式 。 一 般 来 说 ， 多 个 











阐 的 一 种 形式 。 近 年 来 ， 














MySQL 的 读 
每 93 

















， 所 以 没有 特 
到 很 多 高 级 的 
以 及 与 其 他 系 
































' 会 共用 一 台 服 

















的 人 气 越 来 越 高 ， 实 际 上 它 也 拥有 很 多 服务 。 因 为 租借 
赃 器 等 都 是 虚拟 的 ， 所 以 我 们 可 以 根据 数据 量 和 访问 次 数 轻松 地 对 规模 和 规格 进行 修 















































改 。 一 般 按 照 使 用 量 付费 ， 

















云 相关 的 专业 知识 才能 








的 服务 。 





在 很 多 情况 下 有 利 


FF 节约 成 本 。 云 既 包括 一 些 易 于 上 手 的 服务 ， 也 包括 














些 需 要 

















第 10 章 ”使 用 多 个 表 


本 章 将 介绍 处 理 多 个 表 的 方法 。 关 系数 据 库 的 一 个 特征 就 是 
具有 关联 性 。 在 购物 网 站 运行 的 MySQL 数据 库 中 ， 有 数量 惊人 
的 表 在 相互 关联 着 。 
本 章 我 们 将 学 习 如 何 将 表 连 接 起 来 使 用 。 
































































































































10.1 显示 多 个 表 的 记录 


下 面 是 大 家 熟悉 的 员工 信息 表 tbl 和 销售 信息 表 tb， 以 及 在 本 章 中 新 出 现 的 其 他 营业 所 的 员工 
言 息 表 tb2 和 员工 出 生地 信息 表 tb3 。 请 提前 准备 好 表 tb2 和 表 tb3。 
在 本 书 接 下 来 的 内 容 中 ， 将 会 用 到 以 下 各 表 。 




















































































































































































































> 表 tb ( 销售 信息 表 ) > 表 tb1 ( 员工 信息 表 ) > 表 tb2 ( 其 他 营业 所 的 员工 信息 表 ) 
empid sales month empid name age empid name age 
A103 101 4 A101 佐 芯 40 A106 中 村 26 
A102 54 5 A102 高 桥 28 A107 田中 24 
A104 181 4 A103 中 川 20 A108 铃木 23 
A101 184 4 A104 渡 边 23 A109 村 25 
A103 1 5 A105 西 泽 35 A110 吉田 27 
00 > 表 tb3 ( 员工 出 生地 信息 表 ) 
A102 205 6 
A104 93 5 empid region 
A103 12 6 A101 东京 都 
A107 |87 6 A102 震 玉 

A103 神奈川 县 

A104 北海 道 

A105 静 风 
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10.1.2 显示 多 条 提取 结果 


试 着 从 多 个 表 中 取出 记录 ， 并 将 它们 汇总 到 一 起 显示 出 来 。 我 们 可 以 使 用 UNION 命令 从 多 个 
表 中 提取 记录 并 将 它们 合并 起 来 ( 见 图 10-1 )。 


合并 记录 和 列 
列 1 ， 列 2 ， 列 3 ， 
A 1 1 1 

















表 A 
列 1 ， 列 2 ， 列 3 
































A3 
A5 
B1 
B3 
B5 


> 


UNION 


10-1 UNION 
比如 从 去 年 的 客户 表 和 今年 的 客户 表 中 把 老 客 户 的 记录 提取 出 来 ， 然 后 合并 到 一 起 显示 。 也 可 
以 合并 结构 不 同 的 表 的 数据 。 
一 般 来 说 ， 合 并 到 一 起 显示 的 列 ， 其 数据 类 型 需要 一 致 。 不 过 ， 即 使 数据 类 型 不 同 ， 很 多 时 候 
MySQL 也 会 对 可 以 合并 的 记录 进行 合并 。 
使 用 的 语句 非常 简单 ， 只 要 用 UNION 将 “SELECT 列 名 FROM 表 名 ”连接 起 来 即 可 。 



























































将 两 个 表 的 记录 合并 起 来 显示 


; SELECT 列 名 1 FROM 表 名 1 UNION SELECT 列 名 2 FROM 表 名 2; 


写成 一 行 的 话 有 点 不 好 理解 。 执 行 换行 和 缩 进 后 ， 格 式 会 变 成 下 面 这 样 。 
将 两 个 表 的 记录 合并 起 来 显示 ( 换行 + 缩 进 ) 





SELECT 
列 名 1 
FROM 
表 名 1 
: UNION 
SELECT 
列 名 2 
FROM 
表 名 2 ; 
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UNION 将 两 个 “SELECT ..，FROM ...” 的 结果 合并 了 起 来 。 


首先 ， 试 着 合并 具有 相同 列 结构 的 表 tbl 和 tb2 的 记录 。 因 为 合并 的 是 所 有 的 列 ， 所 以 需要 用 
UNION 把 “SELECT * FROM ...” 连 接 起 来 。 


> 执行 内 容 


> 执行 前 ( 表 tb1 ) PF 执行 前 ( 表 tb2 ) 














empid name age je 一 一 合并 tb1 和 tb2 empid name age 
A101 佐 茧 40 
A102 高 桥 28 
A103 中 川 20 中 
A104 渡 边 23 
A105 西 泽 35 






















































































将 表 tb1 和 表 tb2 的 
记录 合并 起 来 显示 

















SELECT * 
FROM tbl 


UNION 
SELECT * 
FROM tb2; 
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执行 结果 

mysql> SELECT * FROM tbl UNION SELECT * FROM tb2; 

ees 虹 要 要 有 本 六 和 + 
empid name age 

Es ES ES 十 
A1l01 佐藤 40 
RA102 高 李 28 
A103 中 川 20 
A104 渡 边 323 
RA105 泽 35 
RA106 中 村 26 
RARA107 田中 24 
Al08 铃木 23 
A109 村 井 25 
RA110 2 

ee es Ee 十 

10 rows in set (0.00 sec) 











为 了 便于 阅读 ， 我们 也 可 以 用 () 把 各 个 “SELECT ...” 括 起 来 。 这 种 写法 更 容易 理解 。 


(SELECT * FROM tb1) UNION (SELECT * FROM tb2); 


上 一 节 ， 我 们 使 用 UNION 合并 了 2 个 表 ， 其 实 ，UNION 也 可 以 一 个 接 一 个 地 合并 多 个 表 。 
将 empid 为 A102、A103、A104、A107 的 记录 一 个 一 个 地 SELECT 出 来 ， 然 后 使 用 UNION 进 
行 合 并 的 示例 如 下 所 示 。 为 了 便于 理解 ， 我 们 添加 了 () 并 进行 了 换行 。 








(SELECT * FROM tb WHERE empid='A102') 
UNION 

(SELECT * FROM tb WHERE empid='A103') 
UNION 


(SELECT * FROM tb WHERE empid='A104') 
UNION 


(SELECT * FROM tb WHERE empid='A107'); 








原来 的 表 中 只 有 empid 是 A101、A102、A103、A104、A107 的 记录 ， 也 就 是 说 ， 我 们 提取 的 
是 除 A101 以 外 的 记录 。 执 行 
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SELECT * 
FROM tb 
WHERE empid 
NOT IN('A101'); 


SELECT * 
FROM tb 

WHERE empid 
IN('A102','A103','A1l104 








也 会 得 到 相同 的 结果 。 需 要 注意 的 是 ， 显 示 的 记录 虽然 相同 ， 但 是 显示 顺序 会 有 所 不 同 。 


B10.14 按 条 件 合并 多 条 提取 结果 进行 显示 


当 使 用 UNION 合并 提取 的 多 条 记录 时 ， 如 果 需 要 添加 条 件 ， 可 以 在 各 个 命令 的 最 后 加 上 
“WHERE 条 件 ”。 

试 着 提取 符合 下 面 两 个 条 件 中 任意 一 个 条 件 的 员工 的 员工 号 (empid )， 并 将 它们 合并 起 来 。 内 
容 有 点 复杂 ， 我 们 来 好 好 确认 一 下 。 

















(DD) 表 tb 中 销售 额 ( sales ) 大 于 等 于 200 万 元 的 员工 的 员工 号 ( empid ) 
@) 表 tb1 中 年 龄 ( age ) 大 于 等 于 35 岁 的 员工 的 员工 号 ( empid ) 





这 么 做 是 为 了 使 用 DD 公司 的 销售 信息 表 tb 选 出 销售 成 绩优 秀 的 员工 ， 再 使 用 员工 信息 表 tb1 
选 出 经 验 丰富 的 老 员 工 。 我 们 可 以 把 它 想象 成 一 份 销售 实力 者 的 名 单 。 

因为 记录 数 较 少 ， 所 以 即使 不 执行 查询 ， 我 们 也 可 以 看 出 符合 条 件 四 的 是 A101 和 Al102， 符 合 
条 件 @) 的 是 Al101 和 A105。 那 么 ,实际 的 抽取 结果 又 是 怎样 的 呢 ? 

我 们 试 着 合并 表 tb 中 sales 大 于 等 于 200 的 员工 的 empid 和 表 tbl 中 age 大 于 等 于 35 的 员工 的 
empid， 然 后 显示 出 来 。 
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区 执行 内 容 























































































































































































































> 执行 前 ( 表 tb ) > 执行 前 ( 表 tb1 ) 
empid | sales month empid | name |age 
A103 |101 4 A101 | 优 芯 ”|40 
A102 |54 5 A102 | 高 桥 |28 
A104 181 4 -7 A103 中 川 20 年 龄 大 于 等 
于 35 悄 
A101 |184 4 A104 | 渡 边 |23 
A103 17 5 A105 | 西 泽 5 
A101 300 5 
1 一 十 销售 额 大 于 等 
A102 205 6 于 200 万 元 
A104 93 5 
A103 12 6 
A107 |87 6 、 4 
> 执行 后 
empid 
合并 表 tb 中 sales 大 于 等 
A101 于 200 和 表 tb1 中 age 大 
A102 等 35 的 员工 的 
empid 并 显示 出 来 
A105 和 2 
操作 方法 
执行 下 面 的 命令 。 
(SELECT 
empid 
FROM 
tb 
WHERE sales>=200) 
UNION 
(SELECT 
empid 
FROM 
tbl 


WHERE age>=35); 
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mysql> (SELECT empid FROM tb WHERE sales>=200) UNION (SELECT empid FROM tbl 
WHERE age>=35) : 


和 学 闷 汪汪 汪汪 中 
| empia | 
Es + 
| Al01 

| Alo2 | 
| Alo5 | 
虽 二 二 + 





3 rows in set (0.00 sec) 








人 10.1.5 合并 显示 多 条 提取 结果 ( 允许 重复 ) 

在 上 一 节 (一 10.1.4 节 ) 中 ， 每 个 员工 的 员工 号 整齐 地 显示 了 出 来 。 大 家 可 能 认为 这 是 理 所 当 
然 的 ， 但 事实 并 非 如 此 。 

这 个 结果 合并 的 是 多 个 表 中 的 记录 。 也 就 是 说 ， 在 10.1.4 节 中 ,根据 条 件 中 提取 了 Al101 和 
Al102， 根 据 条件 @ 提 取 了 Al101 和 A105， 所 以 按理 说 应 该 显示 多 个 A101。 但 是 ，A101 仅 显 示 了 1 
条 记录 。 之 所 以 会 出 现 这 样 的 情况 ， 是 因为 在 提取 记录 的 同时 执行 了 “消除 重复 记录 ”的 操作 。 

如 果 和 本 书 示 例 一 样 记录 的 数量 不 超过 10 条 ,那么 消除 重复 记录 的 操作 就 不 会 造成 什么 影响 ， 
但 如 果 对 大 量 记 录 执 行 “消除 重复 ”的 操作 ， 就 会 产生 一 定 的 等 待 时 间 。 因 此 ， 在 处 理 大 量 记 录 的 
情况 下 ， 省 去 这 项 操作 效率 更 高 。 

我 们 可 以 通过 给 UNION 加 上 ALL 来 省 去 消除 重复 记录 的 操作 。 

把 上 一 节 的 UNION 部 分 换 成 UNION ALL 之 后 ， 就 变 成 了 下 面 这 样 。 

































































操作 方法 





执行 下 面 的 命令 。 

















(SELECT empid FROM tb WHERE sales>=200) 
: UNION ALL 
: (SELECT empid FROM tb1 WHERE age>=35); 
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mysql> (SELECT empid FROM tb WHERE sales>=200) UNION ALL (SELECT empid FROM tbl 
WHERE age>=35); 


4 rows in set (0.14 sec) 

















A101 由 于 涉及 两 个 条 件 的 处 理 ， 所 以 显示 了 两 条 。 








10.2 ”连接 多 个 表 并 显示 ( 内 连接 ) 
上 一 节 介绍 的 UNION 用 于 把 多 个 表 的 记录 合并 在 一 起 。 本 节 将 介绍 内 连接 JOIN 的 相关 内 容 。 
将 多 个 表 通 过 某 个 连接 键 连接 在 一 起 的 处 理 称 为 “连接 ”。 这 种 处 理 符合 关系 数据 库 的 特性 。 
实际 运行 在 Web 上 的 数据 库 通常 是 由 多 个 表 组 成 的 ， 基 本 没有 只 用 一 个 大 表 来 处 理 的 情况 。 


前 面 我 们 操作 了 表 tb 和 tb1。 这 是 一 种 分 开 管 理 公司 “销售 信息 ”和 “员工 信息 ”的 模式 。 我 们 































































































可 以 在 表 也 上 进行 销售 相关 的 处 理 ， 当 需要 用 到 员工 的 姓名 时 再 从 表 tbl 中 把 相关 信息 提取 出 来 。 
> 表 tb ( 销售 信息 表 ) > 表 tb1 ( 员工 信息 表 ) 
empid sales month empid name age 
A103 101 4 A101 佐 茧 40 
A102 54 5 A102 高 桥 28 
A104 181 4 A103 中 川 20 
A101 184 4 A104 渡 边 23 
A103 17 5 A105 西 泽 35 
A101 300 5 
A102 205 6 
A104 93 5 
A103 12 6 
A107 87 6 
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表 了 b 中 empid 为 A101 的 员工 是 40 岁 的 佐 芯 。 可 如 果 只 看 表 世 ， 我 们 是 无 法 得 知 员工 姓名 的 。 
因此 ， 我 们 需要 将 销售 信息 表 tb 和 员工 信息 表 tbl 共同 拥有 的 列 empid 设置 为 连接 键 ， 连 接 这 两 个 

我 们 可 以 使 用 JOIN 连接 两 个 表 。 

连接 两 个 表 


: SELECT 列 名 
FROM 表 1 
: JOIN 要 连接 的 表 2 
: ON 表 1 的 列 = 表 2 的 列 ; 


ON 的 后 面 要 写 上 作为 连接 键 的 列 的 条 件 。 例 如 ， 在 连接 表 tb 和 表 tb1 的 情况 下 ， 由 于 列 empid 
是 共同 的 列 ， 所 以 我 们 要 设置 这 个 列 为 连接 键 。 在 “oN 表 1 的 列 = 表 2 的 列 ”的 部 分 中 ,“x x 
表 的 x x 列 ”的 中 间 要 加 上 “.”， 例 如 tb.empid。 拿 上 面 的 示例 来 说 ， 就 是 下 面 这 样 。 


ON tb.empid=tb]l .empid 


上 面 的 语句 表示 让 “ 表 了 b 的 列 empid” 和 “ 表 tbl 的 列 empid” 作 为 匹配 的 连接 键 。 
下 面 我 们 来 实际 操作 一 下 。 这 里 我 们 暂时 使 用 “*” 将 所 有 的 列 都 显示 出 来 。 





























PP 执行 内 容 




























































































> 执行 前 ( 表 tb ) > 执行 前 ( 表 tb1 ) 
empid sales month empid name age 
A103 101 4 A101 佐 蔷 40 
A102 54 § A102 高 李 28 
A104 181 4 A103 中 川 20 
A101 184 4 m= A104 渡 边 23 
A103 17 5 A105 泽 35 
A101 300 5 

A102 205 6 

A104 93 5 

A103 12 6 

A107 87 6 




















将 列 empid 作 
为 连接 键 
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> 执行 后 
empid sales age 
101 20 
54 28 
181 23 
I 49 连接 表 也 和 表 tb1 
17 20 中 empid 相 匹 配 的 
a00 7 记录 并 显示 出 来 
205 28 
93 23 
人 也 20 
操作 方法 
执行 下 面 的 命令 。 
SELECT * 
FROM tb 
JOIN tbl 





ON tb .empid=tb1l.empid'; 


| 执行 结果 


mysql> SELECT * 


























-> FROM tb 
-> JOIN tbl 
-> ON tb.empid=tbl .empid; 
i Ee 人 Se 人 十 
empid sales month empid name age 
下 入 可 训 和 和 训 扩 和 二 训 这 汪 写 写 这 和 下 演 党 要 近 关 二 二 ES 吉本 有 < + 
RA103 101 4 | A1l03 中 川 20 
A102 54 5 | A102 高 村 28 
ARA104 181 4 | Al04 渡 边 25 
RA101 184 4 | RA101 佐藤 40 
A103 17 5 | A103 中 川 20 
A101 300 5 | Alol 佐藤 40 
A102 205 6 | A102 高 村 28 
ARA104 93 5 | Al04 渡 边 23 
RA103 12 6 | A1l03 中 川 20 
和 开本 这 本 Ss Ss 人 和 3 十 








9 rows in set (0.08 sec) 
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人 内 连接 
通过 结果 可 以 看 到 ， 表 tb 和 表 tbl 中 empid 相 匹 配 的 记录 被 连接 了 起 来 。( 没有 显示 A107 的 


记录 的 原因 请 参考 10.3.1 节 。) 
像 这 样 把 不 同 的 表 中 相 匹 配 的 记录 提取 出 来 的 连接 方式 称 为 内 连接 ( 见 图 10-2 )。 如 果 要 明确 
出 某 一 处 理 是 内 连接 ， 可 以 将 JOIN 部 分 写成 INNER JOIN。 像 下 面 这 样 在 JoIN 的 前 面 加 上 
INNER， 结 果 不 会 发 生 任何 改变 。 








SELECT * FROM tb INNER JOIN tb1l ON tb .empid=tb1l.empid'; 






























































从 另 一 个 表 ] 
提取 值 





















































10-2 ”内 连接 


有 “内 连接 ”的 话 是 不 是 也 有 “外 连接 ” 呢 
与 内 连接 相反 ， 外 连接 (一 10.3.1 节 ) 是 其 中 一 个 表 的 记录 与 其 他 表 的 记录 在 不 匹配 的 情况 下 也 会 提取 出 来 
的 连接 方式 。 















































10.2.2 ”选择 列 进 行 显示 


在 上 一 节 中 ， 我 们 用 “* ”显示 了 所 有 的 列 。 这 次 ,我们 来 选择 要 显示 的 列 。 不 过 ， 在 连接 表 
的 时 候 如 果 只 写 name 和 sales，MySQL 会 不 知道 这 些 列 属于 哪个 表 。 
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在 这 种 情况 下 ， 我 们 要 把 列 名 写成 “ 表 名 . 列 名 ”。 以 表 tb 的 列 sales 为 例 ， 就 是 tb .sales。 
另外 ， 列 会 按照 指定 的 顺序 显示 。 同 一 个 列 显 示 多 少 次 都 没有 关系 。 
例如 ， 在 只 显示 empid、name 、sales 的 情况 下 ，SQL 语句 需要 写成 下 面 这 样 。 








SELECT tb.empid,tbl.name,tb.sales 
FROM tb 


JOIN tbl 
ON tb .empid=tb1l.empid; 














执行 结果 
mysql> SELECT tb .emPid,tbl.name,tb.sales 
-> FROM tb 
-> JOIN tbl 
-> ON tb.empid=tbl .empid; 
ES Ss en 十 
empid name sales 
+------- +------ +------- + 
A103 中 川 下 0 下 
A102 高 桥 54 
Al04 渡 边 181 
A101 估 蔷 184 
A1l03 中 川 17 
A1l01 佐藤 300 
A102 高 桥 205 
A104 渡 边 93 
A1l03 中 川 2 























+------- +------ +------- 十 
9 rows in set (0.02 sec) 











全 10.2.3 给 表 添加 别名 


我 们 也 可 以 给 表 添 加 别名 (一 8.1.3 节 )， 方 法 和 给 列 添加 别名 时 相同 。 
给 表 添 加 别名 的 方法 


下 面 的 命令 用 于 给 表 tb 添加 别名 “x”。 


SELECT * FROM tb AS x; 


给 表 添 加 别名 的 优点 会 在 连接 表 的 时 候 体现 出 来 。 为 了 方便 ， 本 书 使 用 了 tb 这 种 简单 的 表 名 ， 
但 在 实际 工作 中 接触 的 许多 表 都 使 用 了 table_sales250 2 这 种 复杂 的 表 名 。 
当 连 接 多 个 表 时 ， 必 须 按照 “正式 表 名 . 列 名 ”的 方式 进行 记述 。 但 如 果 记 述 的 都 是 table _ 
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sales2 2 .sales 这 样 的 内 容 ， 读 起 来 就 比较 费劲 了 。 

在 这 种 情况 下 ， 如 果 给 表 添 加 一 个 别名 ,就 可 以 将 “正式 表 名 . 列 名 ”写成 “别名 . 列 名 ”。 
比如 table sales2 2.sales 可 以 记述 为 x.sales。 

下 面 我 们 试 着 给 表 添 加 一 个 易于 理解 的 别名 ， 然 后 选择 销售 信息 表 tb 中 的 员工 号 和 销售 额 ， 
以 及 员工 信息 表 tbl 中 的 姓名 进行 显示 。 

先 将 表 了 tb 的 列 empid 与 表 tbl 的 列 empid 相 匹 配 的 记录 进行 连接 ， 然 后 显示 表 tb 的 列 empid、 
表 tbl 的 列 name 和 表 tb 的 列 sales。 但 是 ， 这 些 处 理 要 在 给 表 tb 添加 别名 “x” 和 给 表 tbl 添加 别 
名 “y” 的 前 提 下 进行 。 





> 执行 内 容 
> 执行 前 ( 表 tb (x)) > 执行 前 ( 表 tb1 (y )) 













































































器 | 加 | 四 | 加 | 四 mo 和 | 和 | ol 人 














将 empid 设 置 为 连接 键 














> 执行 后 wv 


中 川 101 

连接 表 tb 和 tb1 中 列 
高 桥 54 eempid 相 匹 配 的 记录 ， 并 
a 将 列 empid、 列 name 和 
渡 力 
渡 边 181 列 sales 显示 出 来 
佐 茧 184 


中 川 17 
佐 芯 300 
高 村 205 
渡 边 93 
中 川 们 
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操作 方法 


执行 下 面 的 命令 。 

SELECT x.empid,y.name,x.sales 
: FROM tb as x 

: JOIN tbl as y 

: ON x.empid=y.empid; 



































执行 结果 
mysql> SELECT x.empid,y.name,x.sales 
-> FROM tb as x 
-> JOIN tbl as y 
-> ON x.empid=y .empid; 
| hs es 十 
empid name sales 
和 i 2 + 
A103 中 川 下 0 于 
RA102 高 桥 54 
A1l04 渡 边 181 
A1l01 佐 茧 184 
RA103 中 川 17 
A1l01 佐 茧 300 
RA102 高 桥 205 
A1l04 渡 边 93 
A103 中 川 12 
洁 二 > 2 + 
9 rows in set (0.00 sec) 











S 10.2.4 ”使 用 USING 使 ON ~ 的 部 分 更 容易 阅读 

在 前 面 的 例子 中 ， 我 们 指定 了 tb .empia = tbl.empid 以 便 将 表示 员工 号 的 列 empid 作为 
连接 条 件 。 在 10.2.3 节 的 示例 中 ， 作 为 连接 键 使 用 的 列 在 两 个 表 中 都 是 empid。 

这 里 列 名 可 以 不 用 相同 ， 只 要 内 容 相同 即 可 。 即 使 表 tb1 的 列 empid 的 列 名 是 empidou 也 没有 
关系 。 在 这 种 情况 下 ， 只 要 写成 ON tb.empid = tbl.empidou 即 可 。 

在 前 面 的 示例 中 ， 作 为 连接 键 的 两 个 列 恰好 都 是 empid 这 个 名 字 。 在 使 用 相同 列 名 进行 指定 的 
情况 下 ， 我 们 可 以 使 用 USING ( 作为 连接 键 的 列 名 ) 简单 地 进行 记述 。 这 时 ，10.2.3 节 中 的 示例 就 
可 以 写成 下 面 这 样 ， 变 得 更 容易 阅读 了 。 
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SELECT tb.empid,tbl.name,tb.sales 
FROM tb 


JOIN tb1 
USING (empid); 





10.2.5 通过 WHERE 设置 条 件 从 连接 表 中 提取 记录 


试 着 通过 WHERE 来 设置 条 件 ， 仅 显示 连接 表 中 符合 条 件 的 记录 。 我 们 只 要 在 最 后 加 上 WHERE 
条 件 即 可 。 

但 是 ， 在 对 表 执 行 连接 的 情况 下 ， 如 果 只 写 “~ WHERE sales>=5”，MySQL 会 不 知道 这 是 
哪个 表 的 empid。 因 此 ， 列 名 一 定 要 按照 “ 表 名 . 列 名 ”的 方式 进行 书写 。 如 果 给 表 设 置 了 别名 
(一 10.2.3 节 )， 也 可 以 记述 为 “别名 . 列 名 ”。 

以 销售 信息 表 tb 中 销售 额 大 于 等 于 100 万 元 的 优秀 成 绩 为 对 象 ， 将 其 与 员工 信息 表 tbl 中 的 姓 
名 进行 合并 然后 显示 出 来 ， 并 给 显示 的 项 目 加 上 别名 。 具 体 来 说 ， 就 是 将 表 tb 和 表 bl 进行 连接 ， 
显示 表 tb 的 列 sales 中 值 大 于 等 于 100 的 记录 。 显 示 列 empid、 列 name 和 列 sales， 并 给 它们 分 别 
加 上 别名 “员工 号 ”“ 姓 名 ”和 “销售 额 ”。 














P 执行 内 容 














































































































执行 前 ( 表 tb ) 执行 前 ( 表 tb1 ) 
empid sales month empid name age 
A103 101 | 4 A101 佐 芯 40 
A102 |54 5 A102 ”| 高 桥 28 
A104 |181 4 si A103 | 中 川 |20 
A101 |184 ela A104 | 波 边 “|23 
A103 17 5 A105 西 泽 35 
A101 300 5 大 0 

A102 205 6 的 值 为 对 象 

Al04 |93 5 

A103 |12 6 

A107 |87 6 




















把 这 两 个 列 作 
为 连接 键 
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执行 后 
员工 号 | 姓名 销售 额 一 一 一 一 显示 别名 
A103 | 中 川 |101 





























和 手表 tb 和 表 tb1 进行 连接 ， 
A101 佐 芯 184 显示 列 sales 中 值 大 于 等 
100 的 记录 
























































操作 方 ; 
执行 下 面 的 命令 。 

















SELECT tb.empid AS 员工 号 ,tbl.name AS 姓名 ,tb.sales RS 销售 额 
FROM tb 

JOIN tbl 
USING (empid) 

WHERE tb.sales>=100; 


























执行 结果 
mysql> SELECT tb.empid AS 员工 号 ,tbl.name AS 姓名 ,tb.sales AS 销售 额 
-> FROM tb 
-> JOIN tbl 
-> USING (emPid) 
-> WHERE tb.sales>=100; 
让 2 再 二 二 二 二 二 二 = + 
员工 号 姓名 销售 额 
Ps 让 下 + 
RA103 中 川 101 
A104 渡 边 181 
A1l01 佐 茧 184 
A1l01 佐 茧 300 
RA102 高 李 205 
下 中 es + 
5 rows in set (0.03 sec) 











S 10.2.6 ”提取 多 个 表 中 的 记录 


我 们 可 以 使 用 JOIN 连接 多 个 表 。 在 “SELECT ... JOIN ... ON ...” 的 基础 上 加 上 “JOIN ... 
ON ...” 就 可 以 对 多 个 表 进 行 连接 。 但 是 ， 在 连接 多 个 表 的 情况 下 ， 人 处 理 时 间 会 相应 地 变 长 ， 记 述 
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的 内 容 也 会 变 得 难以 让 人 理解 。 
对 两 个 或 两 个 以 上 的 表 进 行内 连接 时 需要 使 用 如 下 语句 。 


对 多 个 表 进 行内 连接 


: SELECT ~ FROM 

; 表 名 1 

; JOIN 表 名 2 连接 条 件 
: JOIN 表 名 3 连接 条 件 


D 公司 员工 的 “出 生地 ”信息 保存 在 表 tb3 中 。 如 下 所 示 ， 表 tb3 由 表示 员工 号 的 列 empid 和 





































































































表示 出 生地 的 列 region 构成 。 请 大 家 提前 准备 好 表 tb3。 
> tb3 ( 员工 出 生地 信息 表 ) 
empid region 
A101 东京 都 
A102 和 守 玉 县 
A103 神奈川 县 
A104 北海 道 
A105 静 冈 县 
列 名 empid region 
数据 类 型 VARCHAR (10) VARCHAR (10) 

















员工 号 为 “A101” 的 “ 佐 苹 ”"， 其 出 生地 是 “东京 都 ”。 

表 tb、 表 tbl 和 表 tb3 的 列 empid 中 存在 共同 的 值 。 那 么 我 们 以 列 empid 为 连接 键 ， 试 着 连接 
销售 信息 表 tb、 员 工 信 息 表 tbl 和 员工 出 生地 信息 表 tb3 ， 显 示 员 工 号 ( 表 了 b 的 empid )、 销 售 额 
( 表 妈 的 sales )、 姓 名 ( 表 tbl 的 name ) 和 出 生地 ( 表 tb3 的 region ) 这 4 个 列 。 
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> 执行 内 容 



































































































































































































































> 执行 前 ( 表 tb ) > 执行 前 ( 表 tb1 ) > 执行 前 ( 表 tb3 ) 
sales month name age region 
101 4 佐 艾 40 东京 都 
54 5 高 桥 28 震 玉 县 
181 4 ml 中 川 | 20 神奈川 县 
184 4 渡 边 23 北海 道 
17 5 西 泽 35 静 冈 县 
300 5 
205 6 
93 5 
12 6 4 
87 6 
> 执行 后 
sales name region 
101 中 川 神奈川 县 
54 高 桥 幸 玉 县 
181 渡 边 北海 道 
184 位 苹 ” | 东京 都 连接 表 也 、 表 tb1 和 表 
17 中 川 神奈川 县 tb3， 显 示 列 empid、 列 
300 住 芯 东京 者 sales、 列 name 和 列 region 
205 高 桥 震 玉 县 
93 渡 边 北海 道 
12 中 川 神奈川 县 
操作 方法 
执行 下 面 的 命令 。 
SELECT 
tb.empid,tb.sales,tbl .name,tb3.region 
FROM 
tb 
JOIN 
tb1l 
USING (empid) 
JOIN 
tb3 
USING (empid) 


; 
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mysql> SELECT 
-> tb .empid,tb.sales,tbl .name,tb3.region 
-> FROM 
-> tb 
-> JOIN 
-> tbl 
-> USING (empid) 
-> JOIN 
-> tb3 
-> USING (empid) 





















































































































































于 和 于 和 和 二 各 让 + 
empid sales name region 
+------- 再 二 二 二 二 +------ 可 十 
RA103 101 中 川 神奈川 县 
ARA102 54 高 桥 陵 玉 上 县 
RA104 工 8 江 渡 边 北海 道 
Al01 184 佐 鹏 东京 都 
RA103 17 中 川 神奈川 县 
Al01 300 佐 有 匡 东京 都 
A102 205 高 桥 孙 玉 县 
ARA104 93 渡 边 北海 道 
RA103 TL 中 川 神奈川 县 
+------- 4 $B i + 


9 rows in set (0.00 sec) 











在 连接 条 件 部 分 ， 我们 使 用 3 个 表 共 有 的 列 empid 执行 了 USING (empid)。 如 果 使 用 ON 将 
ON tb.empid=tbl.empid 和 ON tb.empid=tb3.empid 作为 连接 的 条 件 ，SQL 语句 就 需要 写 
成 下 面 这 样 。 


SELECT 

tb.empid,tb.sales,tbl .name,tb3.region 
FROM 

tb 
JOIN 

tb1 


ON tb .empid=tb1l.empid 
JOIN 

tb3 
ON tb .empid=tb3 .empid 


了 
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10.3 显示 多 个 表 的 所 有 记录 ( 外 连接 ) 


S 10.3.1 什么 是 外 连接 






































































































































前 面 我 们 主要 使 用 了 表 tb 和 表 tbl 对 连接 进行 了 说 明 。 我 们 再 来 确认 一 下 这 些 表 的 内 容 。 
> 表 tb ( 销售 信息 表 ) > 表 tb1 ( 员工 信息 表 ) 
empid sales month empid name age 
A103 101 4 A101 佐藤 40 
A102 54 5 A102 高 村 28 
A104 181 4 A103 中 川 20 
A101 184 4 A104 渡 边 23 
A103 17 5 A105 西 泽 35 
A101 300 5 | 

A102 205 6 表 tb 中 没有 

A104 93 5 

A103 12 6 - 

A07 




















在 10.2.3 节 和 10.2.5 节 中 ,我们 使 用 表示 员工 号 的 列 empid 作为 连接 键 对 这 两 个 表 进 行 了 内 
连接 。 

当时 留 下 了 一 个 问题 没有 解决 。 实 际 上 ， 表 tb 中 有 一 个 empid=A107 的 人 ,但 这 个 人 没有 登记 
在 表 tbl 的 员工 名 单 上 。 

10.2.3 节 示 例 的 执行 结果 如 下 所 示 ， 我 们 再 来 确认 一 下 内 容 。 








SELECT x.empid,y.name,x.sales 
FROM tb as x 
JOIN tbl as y 


ON x.empid=y.empid; 





> 表 tb 和 表 tb1 的 内 连接 
































empid name sales 
A103 中 川 101 

| A102 高 桥 54 

| A104 渡 边 181 
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( 续 ) 
empid name sales 
A101 佐 芯 184 
A103 中 川 17 
A101 佐 蔷 300 
A102 高 桥 205 
A104 渡 边 93 
A103 PJ 川 12 




















事实 上 ， 因 为 A107 是 其 他 营业 所 的 员工 ， 所 以 表 tbl1 中 没有 这 个 人 的 数据 。 虽 然 他 贡献 了 一 
部 分 销售 额 ， 但 使 用 JOIN 连接 表 时 不 会 显示 这 个 人 的 记录 。 

另外 ， 员 工 信 息 表 tbl 中 虽然 有 Al105 西 泽 的 信息 ， 但 销售 信息 表 中 没有 他 的 相关 信息 。 也 就 
是 说 ， 虽 然 西 泽 被 列 进 了 员工 名 册 中 ， 但 可 能 因为 没有 销售 成 绩 ， 所 以 销售 信息 表 中 没有 显示 
A105 的 记录 。 也 就 是 说 ， 















































人 使 用 了 JoIN (或 者 INNER JOIN ) 的 “内 连接 ”只 会 提取 与 连接 键 相 匹配 的 记录 


因此 ， 只 存在 于 表 也 或 者 表 tbl 中 的 记录 将 被 忽略 。 
但 是 , 我们 有 时 也 会 遇 到 必须 显示 这 些 记录 的 情况 。 在 这 种 情况 下， 我 们 需要 使 用 外 连接 。 外 
连接 具有 以 下 特征 。 

















人 即使 与 连接 键 不 匹配 ， 外 连接 也 会 提取 另 一 个 表 中 的 所 有 记录 


® 10.3.2 ”外 连接 的 种 类 


根据 连接 时 要 提取 的 是 哪个 表 的 全 部 记录 ， 外 连 A 另外 ， 本 节 用 到 的 图 
是 按照 “SELECT ... FROM 表 1 ... JOIN 表 2 ...” 的 方式 进行 记述 得 出 的 。 





PP 左 外 连接 ( LEFT JOIN ) 
显示 “ 相 匹 配 的 记录 ”和 “ 表 1( 即 左 表 ) 的 全 部 记录 ”。( 见 图 10-3 ) 














P> 右 外 连接 ( RIGHT JOIN ) 
显示 “ 相 匹 配 的 记录 ”和 “要 连接 的 表 2 ( 即 右 表 ) 的 全 部 记录 ”。( 见 图 10-4 ) 
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tb1 ( 要 连接 的 表 2 ) 

empid ，nama ， age 
A101 ， 佐 膝 ， 40 
全 A102 ， 高 桥 ' 28 
A103 ， 中 川 ! 20 
A104 ， 渡 边 ， 23 
A105 ， 西 泽 ， 35 






















































匹配 的 记录 














洲 | 囊 
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所 
贺信 
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CD 
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各 | 驯 
Ea 


A107 仅 存在 于 左 表 中 








倒 

















07 |; NU 


10-3 左 外 连接 





tb ( 表 1 ) tb1 ( 要 连接 的 表 2 ) 
empid : nama ， 
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即使 不 匹配 ， 也 要 提取 
右 表 tb1 的 全 部 记录 























西 泽 仅 存 在 于 右 表 中 


图 10-4 右 外 连接 


S 10.3.3 ”使 用 左 外 连接 


因为 在 销售 信息 表 tb ( 即 左 表 ) 中 有 销售 额 这 一 项 ， 所 以 我 们 试 着 让 员工 信息 表 tbl 中 不 存在 
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的 “Al107” 的 记录 也 显示 出 来 。 
当 进 行 左 外 连接 时 ， 我 们 只 需 将 内 连接 中 使 用 的 JOIN 修改 为 LEFT JOIN 即 可 。 


左 外 连接 





; SELECT 列 名 
FROM 表 1 
: LEFT JOIN 要 连接 的 表 2 
ON 表 1 的 列 = 表 2 的 列 ， 


试 着 将 列 empid 作为 连接 键 ， 使 用 左 外 连接 显示 表 tb 和 表 tpl 相 匹 配 的 记录 ， 以 及 表 tb 的 所 
有 记录 。 但 是 ， 仅 显示 表 也 的 列 empid 和 表 tpl 的 列 name。 


P 执行 内 容 






































































































































> 执行 前 ( 表 tb ( 左 )) > 执行 前 ( 表 tb1 ( 右 )) 
empid | sales month empid name age 
101 4 相 匹 配 40 
54 5 28 
181 4 20 
184 |4 相 匹 配 23 
17 5 A105 西 泽 35 
300 5 = 
205 6 
93 5 
12 6 
A107 87 6 
显示 所 有 记录 
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操作 方法 


执行 下 面 的 命 


少 


口 

SELECT 

tb .empid,tb1l.name 
: FROM 

: tb 


























: LEFT JOIN 
E tb1 
USING (empid) 
执行 结果 
mysql> SELECT 
= 多 tb.empid,tbl .name 
-> FROM 
-> tb 
-> LEFT JOIN 
BE tbl 
-> USING (emPid) 
i 
+------- +------ + 
empid name 
FS 3 + 
A1l01 佐 芯 
云 [0 佐 芯 
RA102 高 桥 
RAR102 高 桥 
RA103 中 
RA103 中 
RA103 中 
RA104 渡 边 
RA104 渡 边 
Al107 NULL 
+------- +------++ 
10 rows in set (0.00 sec) 














执行 结果 中 显示 了 表 tb 和 表 tbl 相 匹 配 的 记录 ， 以 及 左 表 了 tb 的 所 有 记录 。 

只 存在 于 左 表 了 b 中 的 “A107” 的 记录 也 显示 了 出 来 。 男 外 ， 只 存在 于 右 表 tbl 中 的 “A105” 
西 泽 的 记录 没有 显示 出 来 。 

虽然 “A107” 的 记录 被 显示 了 出 来 ,但 由 于 “A107” 在 表 tbl 中 不 存在 相应 的 name， 所 以 
name 为 NULL。 也 就 是 说 ,虽然 显示 了 员工 “A107” 的 记录 ,但 由 于 该 员工 没有 被 记录 在 员工 名 
单 上 ， 所 以 没有 显示 姓名 。 
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C 10.3.4 ”使 用 右 外 连接 


这 次 我 们 试 着 通过 右 外 连接 显示 “ 相 匹 配 的 记录 ”和 “要 连接 的 右 表 的 所 有 记录 ”。 只 要 将 
LEFT JOIN 改 成 RIGHT JOIN 就 可 以 了 。 
本 右 外 连接 


: SELECT 列 名 

: FROM 表 11 

: RIGHT JOIN 要 连接 的 表 2 

ON 表 1 的 列 = 表 2 的 列 ; 

有 些 记录 在 员工 信息 表 tbl 中 有 ,但 在 销售 信息 表 tb 中 没有 ， 对 于 这 种 没有 销售 额 的 员工 的 记 
录 ， 我 们 也 将 其 全 部 显示 出 来 。 

试 着 以 empid 为 连接 键 ， 使 用 右 外 连接 显示 表 tb 和 表 tb1 相 匹 配 的 记录 ， 以 及 表 tb1 的 所 有 记 
录 。 但 是 ， 仅 显示 表 也 的 列 empid 和 表 tb1 的 列 name。 


P 执行 内 容 

















































































































































































































> 执行 前 ( 表 tb ( 左 )) > 执行 前 ( 表 tb1 ( 右 )) 
显示 所 有 记录 
empid sales month age 
4 40 
相 匹 配 28 
4 20 
4 23 
十 人 
5 
6 
5 
6 
A107 87 6 D4 
> 执行 后 
empid | name 
A101 佐藤 
A101 佐 茧 
A102 “| 高 桥 
A102 高 桥 
A103 中 川 
A103 中 川 
A103 中 川 
A104 渡 边 
A104 渡 边 
NULL 洋 
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操作 方法 


执行 下 面 的 命令 。 
SELECT 
tb .empid,tbl .name 





FROM 


: tb 

: RIGHT JOIN 
tb1l 
USING (empid) 









































执行 结果 
mysql> SELECT 
-> tb.empid,tbl .name 
-> FROM 
-> tb 
-> RIGHT JOIN 
= tbl 
-> USING (emPid) 
a 
+------- +------ + 
empid name 
+------- +------ + 
A1l01 佐 蔷 
A1l01 佐藤 
RA102 高 村 
Al102 赎 他 
RA103 中 川 
RAR103 中 川 
RA103 中 川 
RA104 渡 边 
RA104 渡 边 
NULI 泽 
+------- +------ + 
10 rows in set (0.00 sec) 











执行 结果 中 显示 了 表 tb 和 表 tb1l 相 匹 配 的 记录 ， 以 及 右 表 也 的 所 有 记录 。 

只 存在 于 右 表 tbl 中 的 “A105” 西 泽 的 记录 显示 了 出 来 。 另 外 ， 只 存在 于 左 表 tbl 中 的 
“A107” 的 记录 没有 显示 出 来 。 

虽然 “A105” 西 泽 的 记录 显示 了 出 来 ,但 由 于 “A105” 在 表 tb 中 不 存在 相应 的 empid， 所 以 
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empid 为 NULL。 也 就 是 说 ,虽然 显示 了 存在 于 员工 名 单 中 的 西 泽 的 记录 ,但 由 于 西 泽 没有 销售 额 ， 
所 以 没有 显示 销售 信息 表 中 的 员工 号 。 
大 家 需要 弄 清 楚 RIGHT JOIN 和 LEFT JOIN 的 区 别 。 








人 10.3.5 如 避免 混合 使 用 左 外 连接 和 右 外 连接 


当 使 用 外 连接 时 ， 有 一 些 地 方 需要 我 们 注意 。 在 同一 数据 库 中 可 以 混合 使 用 左 外 连接 和 右 外 连 
接 。 另 外 ,我 们 也 可 以 通过 调换 连接 的 表 来 实现 左 外 连接 和 右 外 连接 的 转换 。 

但 是 ,混合 使 用 左 外 连接 和 右 外 连接 可 能 会 导致 日 后 发 生 错 误 。 作 为 数据 库 设 计 的 一 种 技巧 ， 
大 家 要 记 住 不 要 混合 使 用 左 外 连接 和 右 外 连接 。 














加 上 OUTER 后 的 书写 方法 


LEFT JOIN 也 可 以 写成 LEFT OUTER JOIN。 同样 ，RIGHT JOIN 也 可 以 写成 RIGHT OUTER JOIN。 





10.4 ” 自 连 接 


C 10.4.1 什么 是 自 连 接 


我 们 可 以 将 表 与 其 自身 ， 也 就 是 和 同名 的 表 进 行 连接 。 这 种 连接 方式 称 为 自 连 接 。 因 为 是 两 个 
同名 的 表 进行 连接 ， 所 以 如 果 直 接 执行 连接 ， 就 会 显示 出 两 个 同名 的 列 。 这 样 就 无 法 对 列 进行 识别 
(发 生 错 误 )， 因 此 连接 时 必须 定义 别名 (一 8.1.3 节 )。 


自 连 接 









































我 们 可 以 给 同一 个 表 添 加 2 个 别名 ， 即 表 是 1 个 ， 但 名 字 有 2 个 。 但 是 ， 这 样 执行 的 话 ， 会 产 
生 一 些 麻 烦 。 
试 着 对 员工 信息 表 tbl 进行 自 连接 ， 并 把 所 有 的 列 显示 出 来 。 
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区 执行 内 容 








































































































































































































































































































































































































> 执行 前 ( 表 tb1 ) > 执行 前 ( 表 tb1 ) 
empid | name age age 
A101 佐 茧 40 40 
A102 高 桥 28 28 
A103 中 川 20 20 
A104 “| 渡 边 23 23 
A105 西 泽 35 35 
> 执行 后 ( 表 tb1 的 自 连 接 结果 ) 

empid name age name age 

A101 佐 芯 40 40 @ 一 一 一 一 一 所 有 的 组 合 
A102 高 桥 28 40 

A103 中 川 20 40 

A104 渡 边 23 40 

A105 泽 35 40 

A101 左 茧 40 28 

A102 高 村 28 28 

A103 中 川 20 28 

A104 渡 边 23 28 

A105 西 泽 35 28 

A101 估 茧 40 20 

A102 高 桥 28 20 

A103 中 川 20 20 

A104 “| 渡 边 23 20 

A105 泽 35 20 

A101 佐 葬 40 渡 边 23 

A102 高 村 28 渡 边 23 

A103 中 川 20 渡 边 23 

A104 渡 边 23 渡 边 23 

A105 泽 35 渡 边 23 

A101 左 腔 40 西 泽 35 

A102 高 桥 28 西 泽 85 

A103 中 川 20 西 泽 35 

A104 渡 边 23 西 泽 35 

A105 西 泽 35 泽 35 












































10.4 自 连 接 | 205 


操作 方法 





执行 下 面 的 命令 。 




















SELECT * 
FROM tb1 
AS a 
JOIN tb1 
AS b; 








mysql> SELECT * FROM tbl RS a JOIN tbl AS b; 




















































































































































































































Bs 和 二 i 机 < 机 < 十 
empid name age empid name age 
下 出 < 二 再 出 = 再 十 
ARA101 佐 荐 40 | RA101 佐 芯 40 
ARA102 高 桥 28 | Al101 佐 腾 40 
A103 中 川 20 | Al101 佐 芯 40 
RA104 渡 边 23 | Al101 佐 荐 40 
RA105 泽 35 | Al101 佐 有 芋 40 
ARA101 佐 荐 40 | A102 高 桥 28 
A102 高 桥 28 | A1l02 高 桥 28 
A103 中 川 20 | A102 高 桥 28 
RA104 渡 边 23 | A1l02 高 桥 28 
RA105 泽 35 | Al102 高 桥 28 
ARA101 佐 茧 40 | A1l03 中 川 20 
A102 高 桥 28 | A1l03 中 川 20 
A1l03 中 川 20 A1l03 中 川 20 
RA104 渡 边 23 | Al103 中 川 20 
Al05 泽 35 A103 中 川 20 
ARA101 佐 茧 40 | A1l04 渡 边 23 
A1l02 高 桥 28 | A104 渡 边 23 
RA103 中 川 20 | A1l04 渡 边 2 
ARA104 渡 边 23 | A1l04 渡 边 23 
A105 泽 35 | Al04 渡 边 23 
Al01 佐 茧 40 | Al105 泽 35 
A102 高 桥 28 | Al105 泽 35 
RA103 中 川 20 A1l05 泽 35 
RA104 渡 边 23 | Al05 泽 35 
RA105 泽 35 Al05 泽 35 
下 二 = 于 本 二 二 十 


25 rows in set (0.01 sec) 











206 | 第 10 章 使 用 多 个 表 




















由 于 表 tb1 是 与 不 同名 的 自身 进行 了 连接 ， 所 以 自身 拥有 的 所 有 记录 将 与 自身 拥有 的 所 有 记录 
进行 连接 。 

“ 佐 功 ”的 记录 中 会 加 上 “ 佐 茧 ”“ 高 桥 ”“ 中 川 ”" 渡 边 ”“ 西 泽 ” 的 记录 ,“ 高 桥 ” 的 记录 中 会 
加 上 “ 优 蕨 ”“ 高 桥 "”“ 中 川 ”“ 渡 边 ”“ 西 泽 ” 的 记录 ， 像 这 样 记录 数 x 记录 数 ， 连 接 的 记录 数 就 
会 变 得 非常 庞大 。 这 个 示例 中 进行 自 连接 的 表 有 5 条 记录 ， 结 果 共 有 5x5=25 条 记录 。 如 果 表 中 
的 记录 超过 了 1000 条 ， 那 么 使 用 自 连 接 就 不 合适 了 。 
其 实在 数据 库 领 域 ， 这 种 使 用 “ 蛮 力 ”的 方法 ， 正 在 被 理所当然 地 使 用 着 。 







































































10.4.2 ”排序 的 技巧 其 一 

将 两 个 完全 相同 的 表 进 行 连接 究竟 能 够 实现 什么 ”至 少 自 连接 的 结果 中 会 包含 所 有 的 组 合 。 如 
果 其 中 有 你 想 要 的 组 合 ， 之 后 就 可 以 通过 设置 条 件 来 选 出 想 要 的 内 容 了 。 

这 里 介绍 一 个 使 用 自 连 接 的 典型 示例 一 一 排序 。 如 果 你 是 数据 库 的 初学 者 ， 可 能 会 觉得 排序 很 
简单 。 但 遗憾 的 是 ， 数 据 库 中 没有 像 Excel 的 RANK 这 样 的 函数 。 

数据 库 中 的 排序 其 实 很 麻烦 ， 必 须 通 过 组 合 使 用 ORDER 和 GROUP 等 关键 字 来 完成 处 理 。 









































区 自 连接 的 验证 


我 们 试 着 按照 年 龄 从 大 到 小 的 顺序 对 员工 信息 表 tbl 中 的 记录 进行 排名 。 
我 们 再 来 确认 一 下 10.4.1 节 中 自 连接 的 结 


> 表 tb1 的 自 连 接 结果 ( 节选 ) 
















































































































































































empid name age empid name age 
只 有 这 一 
个 (第 一 ) A1 
A1 
A1 等 于 40 
A1 
A1 
第 一 A 
第 三 A1 
A1 和 于 28 
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请 看 右 侧 为 “ 佐 蕨 40” 左 侧 为 “ 佐 蕨 40”“ 高 桥 28”“ 中 川 20”“ 渡 边 23”“ 西 泽 35” 的 5 行 
记录 。 在 这 5 行 记录 中 ， 大 于 等 于 右 侧 “佐藤 40” 中 40 这 个 数字 的 数据 在 左 侧 仅 有 1 个 ， 所 以 佐 
茧 排名 第 一 。 

同样 ， 请 看 右 侧 为 “高 桥 28”、 左 侧 为 “ 佐 芯 40”“ 高 桥 28”…… 的 5 行 记录 。 其 中 ， 大 于 等 
于 右 侧 “高 桥 28” 中 28 这 个 数字 的 数据 在 左 侧 有 3 个 ， 分 别 为 40、35、28 ， 也 就 是 说 ， 高 桥 排 在 
第 三 。 

像 这 样 ， 在 右 侧 相同 的 5 行 记 录 的 范围 内 ， 大 于 等 于 右 侧 age 的 值 在 左 侧 age 中 的 个 数 就 是 
排名 。 

换 名 话说， 就 是 只 需要 进行 自 连 接 ， 并 对 每 一 组 empid 计算 大 于 等 于 右 侧 age 的 值 在 左 侧 
age 中 的 个 数 即 可 。 该 数 可 以 通过 COUNT (* ) 计算 。 我 们 可 以 把 它 当 成 一 个 如 何 编 写 SQL 语句 的 
提示 。 





























语句 基本 上 就 是 10.4.1 节 中 执行 过 的 “SELECT * FROM tbl AS a JOIN tbl AS bi; ”。 
请 大 家 认真 思考 之 后 再 阅读 后 面 的 答案 。 
下 面试 着 从 表 tbl 中 age 较 大 的 记录 开始 进行 排名 ， 并 显示 name 、age 和 排名 这 3 个 列 。 


全 表 的 结构 
























































































































































> 执行 前 ( 表 tb1 ) > 执行 前 ( 表 tb1 ) 
empid name age empid name age 
A101 佐 爱 40 8 一 一 一 一 自 连接 一 一 一 ®| A101 佐 爱 40 
A102 高 桥 28 A102 高 桥 28 
A103 中 川 20 rls A103 中 川 20 
A104 渡 边 23 A104 渡 边 23 
A105 泽 35 A105 泽 35 
D4 

> 执行 后 

name age COUNT (*) 

佐 芯 40 








8 一 根据 age 显示 的 排名 结 















































J 
U 
ee 
DD 
[= 
六 | 人 | 加 | 一 
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操作 方法 
执行 下 面 的 命令 。 




















: SELECT a.name,a.age,COUNT (*) 
FROM tbl AS a 

， JOIN tbl AS b 

WHERE a.age<=b.age 
GROUP BY a.empid; 



































执行 结果 
mysql> SELECT a.name,a.age,COUNT (*) 
-> FROM tbl AS a 
-> JOIN tbl AS Pb 
-> WHERE a.age<=b .age 
-> GROUP BY a.empid; 
+------ +------ 半生 + 
name age COUNT (*) 
+------ +------ +---------- + 
佐 茧 40 1 
高 村 28 3 
中 川 20 5 
渡 边 23 4 
西 泽 35 2 
+------ +------ SR 十 
5 rows in set (0.05 sec) 











首先 ， 因 为 是 自 连接 ， 所 以 使 用 的 语句 是 “SELECT * FROM tbl AS a JOIN tbl AS 
b;”。 对 其 设置 WHERE 条 件 a.age <=b.age 之后, 每 组 a.empid 中 大 于 等 于 a.age 的 b.age 
记录 就 会 被 提取 出 来 ， 之 后 再 用 couNT (* ) 计算 记录 个 数 就 会 得 出 排名 。 














10.5 从 SELECT 的 记录 中 SELECT ( 子 查询 ) 


10.5.1 什么 是 子 查询 


使 用 子 查询 可 以 完成 两 个 阶段 的 处 理 : 执行 查询 ， 然 后 使 用 检索 到 的 记录 进一步 执行 查询 。 例 如 ， 在 
第 一 阶段 中 进行 “从 销售 信息 表 中 提取 销售 额 大 于 等 于 200 万 元 的 员工 号 ”的 处 理 ， 然 后 在 第 二 阶段 中 
进行 “从 提取 出 来 的 员工 号 中 提取 相对 应 的 姓名 ”的 处 理 ( 见 图 10-5 ) 第 一 阶段 的 查询 称 为 子 查 询 。 











第 一 阶段 的 子 查 询 可 以 返 

















10.5 从 SELECT 的 记录 


回 值 、 列 和 记录 等 。 





许多 使 用 了 子 查询 的 处 理 可 以 用 其 他 方法 取代 ， 例 如 内 连接 。 但 是 ， 





理解 ， 处 理 效率 也 更 高 。 








10.5.2 ”显示 最 大 值 ( 














\ 
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子 查询 的 处 理 方式 更 容易 



























提取 销售 额 大 于 科 




















F200 万 元 的 记录 | 






























































10-5 子 查 询 


反 回 值 的 子 查询 其 一 ) 


过 


一 一 一 一 一 一 一 一 一 一 一 一 





下 面 这 种 情况 该 如 何 处 理 呢 ? 


DS 显示 表 tb 中 列 sales 最 大 值 的 记录 








换 名 话说， 就 是 显示 在 销售 信息 表 tb 中 销售 额 最 高 (sales 最 大 ) 的 员工 记录 。 列 的 最 大 值 可 
以 使 用 MAX 函数 计算 。 我 们 来 看 一 下 表 也 的 结构 ， 思 考 一 下 如 何 处 理 。 









































上 表 tb 

empid sales month 
A103 101 4 
A102 54 5 
A104 181 4 
A101 184 4 
A103 17 5 
A101 300 5 
A102 205 6 
A104 93 5 
A103 12 6 
A107 87 6 
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或 许 有 人 会 想 :“ 这 还 不 简单 ， 按 照 下 面 的 方式 处 理 不 就 可 以 了 。” 


SELECT * FROM tb WHERE sales=MAX (sales); 


试 着 执行 一 下 就 知道 了 ， 结 果 会 发 生 错 误 。 


执行 结果 





mysql> SELECT * FROM tb WHERE sales=MAX (sales); 
ERROR 1111 (HY000): Invalid use of group function 











只 写 MAX (sales) 是 不 会 计算 出 列 sales 的 最 大 值 的 。 想 要 获取 列 sales 中 的 最 大 值 ， 就 需要 
按照 下 面 的 方式 进行 查询 。 


SELECT MAX (sales) FROM tb 


中 


当 提 取 最 大 值 相对 应 的 记录 时 ， 需 要 先 通过 上 面 的 操作 获取 最 大 值 ， 然 后 从 表 tb 中 提取 
记录 。 

这 里 ， 子 查询 就 起 到 了 非常 重要 的 作用 。 首 先 ， 在 第 一 阶段 的 查询 中 取出 MAX (sales) 的 值 ， 
然后 在 第 二 阶段 选择 ( SELECT ) 包含 了 最 大 值 的 列 sales 的 记录 ( 见 图 10-6 )。 


SELECT * FROM tb WHERE sales IN (第 一 阶段 的 处 理 结果 ) 
i 


SELECT Se SELECT 


FROM 
Wiel sa | < 一 一 MAX(sales) 


IN FROM tb; 





第 二 阶段 第 一 阶段 ( 子 查询 ) 
10-6 子 查询 的 流程 
表示 和 条件“ 包含 x x” 的 “WHERE ... IN” 的 使 用 方法 请 参考 8.3.3 节 。 相 当 于 第 一 阶段 处 理 
的 子 查 询 需 要 使 用 () 括 起 来 。 注 意 ， 忘 记 加 () 会 发 生 错误 。 
我 们 来 实际 操作 一 下 。 试 着 显示 表 tb 中 列 sales 最 大 值 的 记录 。 
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PP 执行 内 容 








































































































> 执行 前 ( 表 tb ) > 执行 后 
empid sales month ， empid sales month 
A103 101 4 A101 300 5 居 一 一 一 提取 并 显示 
A102 54 5 
A104 181 4 
A101 184 4 
A103 17 5 
A101 300 已 
A102 205 6 
A104 93 5 
A103 人 6 
A107 87 6 
此 列 中 二 
操作 方法 




















SELECT * 
FROM tb 
WHERE sales 
IN (SELECT MAX(sales) FROM tb); 








执行 结果 
mysql> SELECT * 

= FROM tb 

-> WHERE sales 

= IN (SELECT MAX (sales) FROM tb); 
+------- +------- +------- 十 


| empid | sales | month | 


1 row in set (0.28 sec) 
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GROUP BY 函数 


MAX、AVG 和 SUM ( 一 8.2.2 节 ) 等 聚合 函数 也 称 为 “GROUP BY 函数 "， 这 类 函数 用 了 


是 ， 在 没有 “GROUP BY …” 的 情况 下 ， 这 类 函数 会 将 整个 表 作为 一 个 组 进行 处 理 。 








在 上 一 节 中 ， 我 们 介绍 了 使 用 MAX 函数 的 子 查 询 。 





























处 再 





分 组 


S 10.5.3 ”提取 大 于 等 于 平均 值 的 记录 ( 返回 值 的 子 查 询 其 二 ) 





算 员工 信息 表 tbl 中 员工 的 平均 年 龄 ， 并 提取 大 于 等 于 平均 年 龄 的 员工 的 记录 。 


在 第 一 阶段 ， 
等 于 这 个 平均 值 的 员工 的 记录 。 























我 们 需要 使 用 AVG 函数 计算 表 tb1 























下 面试 着 计算 表 tbl 中 列 age 的 平均 值 ， 并 显示 年 龄 大 于 等 于 这 个 值 的 员工 的 记录 。 


P 执行 内 容 
> 执行 前 ( 表 tb1 ) 






























































后 的 值 。 但 


下 面 再 来 看 一 个 使 用 函数 的 示例 。 试 着 计 


! 列 age 的 平均 值 ， 然 后 在 第 二 阶段 提取 大 于 



































> 执行 后 

empid name age 

A101 佐 葬 40 只 显示 大 于 等 
一 | 于 平均 值 的 员 

A105 1 35 工 的 记录 















































empid name age 

A101 佐 茧 40 

A102 高 桥 28 

A103 中 川 20 

A104 渡 边 28 

A105 泽 35 

计算 此 列 的 平均 值 
操作 方法 
执行 下 面 的 命令 。 





























SELECT * 
FROM tb1 
: WHERE age 
: >= (SELECT AVG(age) FROM tb1); 
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mysql> SELECT * 

= FROM tbl 

-> WHERE age 

-> >= (SELECT AVG(age) FROM tb1); 
2 Re re + 
| empiqd | name | age | 
二 机 下 十 
| Alol | 佐藤 | 40 | 
| A1l05 | 泽 | 35 | 
下 a 3 十 
2 rows in set (0.06 sec) 


























表 tbl 中 员工 的 平均 年 龄 是 (40 + 28 + 20 +23 +35) <5， 即 29.2。 因 此 ， 大 于 29.2 的 40 岁 
的 佐 蕊 和 35 岁 的 西 泽 的 信息 显示 了 出 来 。 





已 10.54 使 用 IN (返回 列 的 了 查询 ) 


下 面 是 使 用 子 查询 返回 列 的 示例 。 我 们 要 在 第 一 阶段 的 子 查询 中 返回 符合 条 件 的 列 ， 然 后 在 第 
二 阶段 中 提取 包含 该 列 的 记录 。 在 这 种 情况 下 ， 我 们 需要 像 下 面 这 样 使 用 IN。 
子 查询 的 语句 


: SELECT 显示 的 列 FROM 表 名 
WHERE 列 名 IN ( 通过 子 查询 SELECT 语句 提取 的 列 ) ; 


第 一 阶段 的 查询 〈 子 查询 ) 必须 使 用 () 括 起 来 。 先 执行 第 一 阶段 的 提取 操作 ， 然 后 以 列 中 包 
含 该 值 为 条 件 ( IN )， 执 行 第 二 阶段 的 提取 操作 。 下 面 我 们 来 处 理 以 下 内 容 。 






































2 显示 销售 额 大 于 等 于 200 万 元 的 员工 姓名 














具体 的 处 理 方法 如 下 。 








2 提取 销售 信息 表 tb 中 销售 额 ( sales ) 大 于 等 于 200 的 empid， 然 后 显示 员工 信息 表 tb1 中 
该 员工 的 姓名 ( name ) 


第 一 阶段 子 查 询 实现 的 内 容 是 提取 表 tb 中 满足 条 件 sales >=200 的 记录 的 empid。 具 体 执行 
的 命令 如 下 所 示 。 





SELECT empid FROM tb WHERE sales>=200; 
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这 个 查询 对 我 们 来 说 并 不 陌生 。 执 行 该 命令 会 提取 出 “A101” 和 “Al102” 的 记录 。 然 后 在 第 
二 阶段 ， 从 表 tbl 中 提取 出 第 一 阶段 提取 的 empid 相应 的 记录 。 


SELECT * FROM tbl WHERE 条 件 


“和 条件” 部 分 指 的 是 包含 第 一 阶段 中 提取 的 “Al101” 和 “Al102” 的 记录 。 我 们 需要 使 用 
“WHERE ... IN .. 


试 着 使 用 子 查询 提取 表 tb 中 sales 大 于 等 于 200 的 empid， 然 后 显示 表 tbl 中 相应 的 记录 。 








9? 




















> 执行 内 容 































































































































































































> 执行 前 ( 表 tb ) > 执行 前 ( 表 tb1 ) 
empid sales month empid name age 
A103 101 4 A101 佐藤 40 
[H 提取 相应 的 记录 
A102 54 5 A102 高 桥 28 
A104 181 4 A103 中 川 20 
A101 184 4 可 = A104 “| 渡 边 23 
A103 Wy 5 A105 泽 35 
A101 300 5 3 
中 -一 一 提取 值 大 于 等 
A102 205 6 于 200 的 记录 
A104 93 5 
A103 12 6 
A107 87 6 y 
> 执行 后 
empid name age 
提取 sales 大 于 等 
和 9 估 茧 人 200 的 empid， 并 显示 
A102 高 桥 28 表 tb1 中 相应 的 记录 
操作 方法 
执行 下 面 的 命令 。 
SELECT * 
FROM tb1l 


WHERE empid 
IN (SELECT empid FROM tb WHERE sales>=200); 
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mysql> SELECT * 

= FROM tbl 

-> WHERE empid 

= IN (SELECT empid FROM tb WHERE sales>=200) : 
Fe Re 2 十 
| empiqd | name | age | 
4 二 下 十 
| Alol | 佐藤 | 40 | 
| A102 | 高 桥 | 28 | 
ee 2 十 
2 rows in set (0.01 sec) 

















和 我 们 预想 的 一 样 ,，“A101” 和 “A102” 显 示 了 出 来 。 





子 查询 和 内 连接 的 提取 结果 的 差异 ( 虽然 相似 但 不 同 ! ) 


子 查询 和 内 连接 ( 一 10.2 节 ) 非常 相似 。 我 们 来 看 一 个 例子 。 对 于 销售 信息 表 tb 中 存在 的 员工 号 empid， 
显示 员工 信息 表 tb1 中 相应 的 员工 号 empid 和 姓名 name。 下 面 是 分 别 用 “ 子 查询 ”和 “内 连接 ”实现 上 述 内 容 
的 





















































E SELECT empid,name 

9 FROM tb1 

: WHERE empid 

IN (SELECT empid FROM tb) ; 





: SELECT tbl.empid,tbl.name 
FROM tb1l 

: JOIN tb 

ON tbl.empid=tb.empid; 














二 者 的 执行 结果 是 不 同 的 。 在 使 用 子 查询 的 情况 下 ， 会 先 提取 表 tb 中 存在 的 empid， 然 后 仅 显示 和 表 tb1 
相 匹 配 的 记录 。 而 使 用 内 连接 的 话 ， 将 显示 表 tb 中 所 有 的 记录 。 
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二 尖子 查询 





mysql> SELECT empid,name 





-> FROM tbl 
-> WHERE empid 
-> IN (SELECT empid FROM tb); 
+------- +------ 十 
| empid | name | 
时 二 二 三 二 二 二 十 
| 0 | 证 | 
| A102 | 高 桥 | 
| A103 | 品川 | 
| A104 | 渡 边 | 
Ps 二 二 三 二 二 十 








mysql> SELECT tbl .empid,tbl .name 


























=> FROM tb1 
-> JOIN tb 
三 至 ON tbl .emPid=tb .emPid: 
OO 二 三友 十 
empid name 
证 二 二 三 十 
RA103 中 川 
OS 高 桥 
A104 渡 边 
mao 佐藤 
RA103 中 川 
2 佐藤 
ioZ 高 桥 
A104 渡 边 
A1l03 中 川 
下 三 二 二 三 三 下 三 二 二 二 三 三 十 


9 rows in set (0.00 sec) 


























但 是 ， 在 内 连接 中 ， 如 果 在 SELECT 之 后 加 上 DISTINCT (一 8.3.5 节 )， 则 会 提取 与 子 查询 相同 的 记录 。 








C 10.5.5 ”使 用 “=” 代 替 IN 会 报错 吗 


有 人 可 能 觉得 在 上 一 节 的 示例 中 可 以 像 下 面 这 样 使 用 “=” 来 代 蔡 IN。 
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SELECT * 
FROM tb1l 


WHERE empid = (SELECT empid FROM tb WHERE sales>=200); 





很 遗憾 ， 执 行 此 命令 会 出 现 supquery returns more than 1 row ( 子 查询 的 返回 结 
多 于 一 行 ! ) 的 错误 。 如 果 满 足 “empid 与 x x 恰好 一 致 ”的 条 件 ， 使 用 “=” 也 无 妨 。 但 是 在 这 
个 示例 中 ， 多 条 记录 在 第 一 阶段 被 提取 了 出 来 ， 因 此 必须 使 用 表示 “…… 之 一 ”的 IN。 

当然 ， 如 果 只 有 一 条 相对 应 的 记录 ， 那 么 即使 使 用 “=” 也 不 会 报错 。 例 如 ， 下 面 是 使 用 
LIMIT 提取 一 条 记录 的 示例 。 这 么 做 至 少 可 以 避免 报错 。 





回 
汀 




















SELECT * 
FROM tb1l 
WHERE empid 
= (SELECT empid FROM tb WHERE sales>=200 LIMIT 1); 








但 是 这 么 一 来 ,我 们 就 无 法 得 知 提取 的 是 “A101” 还 是 “A102” 了 ,所 以 LIMIT 在 这 里 并 没 
有 起 到 太 大 的 作用 。 

我 们 可 以 降序 排列 表 中 记录 ， 然 后 只 让 第 一 条 记录 显示 出 来 。 下 面 是 使 用 ORDER BY 以 “ 销 
售 额 最 高 的 员工 ”为 条 件 提取 记录 的 示例 。 在 第 一 阶段 ， 查 询 表 tb 中 sales 的 最 大 值 所 对 应 的 
empid， 然 后 在 第 二 阶段 从 表 tb1 中 提取 相应 的 记录 。 


SELECT * 
FROM tb1l 


WHERE empid 
= (SELECT empid FROM tb ORDER BY sales DESC LIMIT 1); 





mysql> SELECT * 





-> FROM tbl 

-> WHERE empid 

-> =(SELECT empid FROM tb ORDER BY sales DESC LIMIT 1); 
+------- +------ +------ 十 


| empid | name | age | 
+------- +------ +------ 十 
| Alol | 佐藤 | 40 | 


玉 = +------ +------ 十 





1 row in set (0.06 sec) 








当 使 用 LIMIT 1 时 仅 能 提取 一 条 记录 ， 所 以 使 用 “=” 也 不 会 报错 。 
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S 10.5.6 ”使 用 EXISTS， 仅 以 存在 的 记录 为 对 象 


我 们 在 10.5.4 节 中 学 习 的 是 “WHERE ... IN (SELECT ...)” 这 种 类 型 的 子 查 询 。 它 用 于 返回 
符合 第 一 阶段 子 查 询 结果 的 列 的 记录 。 例 如 ， 在 10.5.4 节 的 示例 中 返回 了 Al101 和 Al102 所 在 的 列 
的 数据 ， 之 后 又 以 包含 其 值 的 记录 为 对 象 进行 了 提取 。 

我 们 也 可 以 使 用 ExISTS， 不 返回 指定 的 列 而 返回 “第 一 阶段 子 查询 中 存在 目标 记录 ”这 样 的 信息 。 

用 语言 描述 不 容易 理解 ， 我 们 来 看 一 个 实际 的 例子 。 员 工 信 息 表 tpl 中 也 包含 了 销售 信息 表 tb 
中 没有 的 员工 号 的 记录 ， 即 “A105” 的 记录 。 








































































































> 表 tb1 ( 员工 信息 表 ) > 表 tb ( 销售 信息 表 ) 
empid name age empid | sales month 
A101 佐 茧 40 Al103 “| 101 4 
A102 “| 高 桥 ”|28 THEA0 [ms 5 
A103 | 中 川 |20 Al04 |181 |4 
A104 | 渡 边 “| 23 Al01 |184 |4 
A105 ”| 西 泽 ”|35 ”le 一 一 | 没有 销售 客 aios 7 5 
A101 |300 |5 
A102 |205 6 
A104 |93 5 
A103 |12 6 
A107 |87 6 

















也 就 是 说 ,“A105” 的 西 泽 没 有 销售 额 。 
下 面试 着 使 用 子 查 询 显示 “员工 信息 表 上 有 销售 额 的 员工 的 记录 "。 具 体 来 说 ， 就 是 进行 如 下 
处 理 。 














2 从 表 tb 中 提取 有 销售 额 的 员工 的 记录 ， 然 后 从 表 tb1 中 提取 相对 应 的 记录 显示 出 来 








首先 ， 因 为 要 显示 的 是 员工 信息 表 tbl 的 记录 ， 所 以 要 执行 如 下 命令 。 


SELECT * FROM tb1 


其 次 ,“ 有 销售 额 的 员工 ”是 指 该 员工 的 empid 在 表 tb 和 表 tbl 中 都 存在 ， 所 以 需要 添加 如 下 


WHERE tb.empid=tbl .empid 


从 表 也 中 选择 符合 上 述 条 件 的 记录 作为 第 一 阶段 的 处 理 。 
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SELECT * FROM tb WHERE tb.empid=tbl.empid 


这 次 提取 的 并 不 是 指定 的 列 ， 而 是 符合 条 件 的 记录 。 使 用 EXISTS 从 表 tbl 中 提取 相应 的 记录 
作为 第 二 阶段 的 处 理 ， 最 终结 果 如 下 所 示 。 








操作 方法 





执行 下 面 的 命令 。 

















: SELECT * 

: FROM tbl 

| WHERE EXISTS 

| (SELECT * FROM tb WHERE tb.empid=tbl1 .empid); 








mysql> SELECT * 
SS FROM tbl 
-> WHERE EXISTS 
二 法 (SELECT * FROM tb WHERE tb .emPid=tbl.emPid) : 
+------- +------ 业 十 
empid | name | age | 
+------- +------ +------ 十 
Al01 | 佐藤 | 40 | 
Al02 | 高 桥 | 28 | 
A103 | 中 川 | 20 | 
Al04 | 渡 边 | 23 | 
+------- +------ +------ 十 
4 rows in set (0.00 sec) 











A105 在 表 tb 中 不 存在 相应 的 empid， 所 以 没有 显示 出 来 。 从 中 我 们 可 以 知道 ，A105 的 西 泽 没 
有 销售 额 。 





CS 10.5.7 NOT EXISTS 


相反 ，NOT EXISTS 以 子 查询 不 提取 的 记录 为 对 象 进行 处 理 。 那 么 我 们 来 试 着 使 用 子 查 询 显 示 
“没有 销售 额 的 员工 ”的 记录 。 

这 里 只 是 将 EXISTS 改 成 了 NOT EXISTS。 试 着 使 用 NOT EXISTS 提取 表 tb 中 不 存在 的 
empid， 并 显示 表 tb1 中 相应 的 记录 。 
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> 执行 内 容 
































































































































> 执行 前 ( 表 tb1 ) > 执行 前 ( 表 tb ) 
empid | name age empid | sales month 
A101 | 傣 茧 | 40 0 4 | | 个 在 此 处 
A102 高 桥 28 加 晶 A102 54 5 
A103 中 川 20 A104 181 4 
A104 渡 边 23 A101 184 4 
A105 西 泽 35 A103 17 5 
A101 300 5 
A102 205 6 
A104 93 5 
A103 12 6 
vw A107 87 6 
> 执行 后 
empid | name age 
EE 


















































SELECT * 
FROM tbl 
WHERE NOT EXISTS 
(SELECT * FROM tb WHERE tb.empid=tbl .empid); 





mysql> SELECT * 

-> FROM tbl 

-> WHERE NOT EXISTS 

-> (SELECT * FROM tb WHERE tb.empid=tbl .empid); 
+------- +------ +------ + 


| empiqd | name | age | 

















+------- +------ +------ + 
| A105 | 西 泽 | 35 | 
+------- +------ +------ 十 
1 row in set (0.00 sec) 











1 
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从 中 我 们 可 以 知道 ， 没 有 销售 额 的 员工 是 35 岁 的 A105 西 泽 。 





0.5.8 排序 的 技巧 其 二 


一 人 一 一 一 一 一 一 一 一 一 


非常 


的 方 


中 。 


询 的 


rank, 





对 电子 表格 软件 (Excel ) 来 说 ， 排 序 是 一 件 很 简单 的 事 。 但 是 在 RDBMS 中 ， 排 序 却 是 一 个 
css 这 一 点 我 们 在 10.4.2 节 的 自 连 接 中 也 介绍 过 。 

， 在 使 用 子 查 询 的 情况 下 ， 我 们 可 以 用 各 种 各 样 的 方法 进行 排序 ， 而 且 这 些 方法 比 自 连接 
se 这 里 来 介绍 其 中 一 种 方法 。 
试 着 对 销售 信息 表 tb 的 销售 额 进行 排名 。 这 次 我 们 要 准备 其 他 的 表 ， 并 试 着 把 排名 输入 到 列 
内 容 有 些 复杂 ， 我 们 需要 静 下 心 来 一 步 一 步 地 操作 ， 具 体 的 处 理 思路 如 下 所 示 。 





























向 具有 自动 连续 编号 功能 的 表 中 插入 按照 sales 由 高 到 低 的 顺序 排列 的 记录 
v 
自动 输入 的 连续 编号 就 是 排名 


























具体 来 说 ， 就 是 按照 下 面 的 步 又 进行 处 理 。 














创建 和 表 tb 结构 相同 的 表 tb_rank 
vv 

向 表 tb_rank 中 添加 具有 自动 连续 编号 功能 的 列 c_rank 
vv 

对 表 tb 执行 按 列 sales 由 高 到 低 排 序 的 SELECT 子 查询 
vy 

将 子 查询 的 结果 INSERT 到 表 tb_rank 中 


向 表 tb_rank 的 列 c_rank 中 输入 排名 。 

那么 ,用 于 输入 排名 的 查询 语句 具体 应 该 怎么 写 呢 ? 请 大 家 思考 一 下 自动 连续 编号 功能 和 子 查 
使 用 方法 。 
表 tb 是 由 empid、sales 和 month 这 3 个 列 组 成 的 。 我 们 在 此 基础 上 添加 列 c_rank 创建 表 tb_ 
并 复制 表 了 b 的 记录 到 该 表 中 。 然 后 试 着 向 列 c_rank 中 输入 sales 的 排名 。 
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区 执行 内 容 


> 执行 前 ( 表 tb ) 


列 empid sales month 























数据 类 型 ”| VARCHAR (10) | INT INT 


b4 


基 复制 表 tb， 创 建 表 tb_rank 


列 empid sales month ”8@ 一 一 一 一 一 复制 列 结构 























数据 类 型 ”| VARCHAR (10) | INT INT 




















> 向 表 tb_rank 中 添加 具有 自动 连续 编号 功能 的 列 c_rank 



















































































列 empid sales | month 
数据 类 型 ”| VARCHAR (10) INT INT 添加 具有 自 
动 连续 编号 
功能 的 列 
> 对 表 tb 执行 按 列 sales 由 高 到 > 向 表 tb_rank 的 列 c_rank 中 输入 排名 
低 排 序 的 SELECT 子 查询 empid sales month c_rank 
A101 300 5 
A102 205 6 
A101 184 4 
A104 181 4 
A103 101 4 
4 A104 93 5 
A107 87 6 
A102 54 5 
A103 17 5 
A103 12 6 
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操作 方法 






































ALTER TABLE tb rank ADD C_rank INT AUTO INCREMENT PRIMARY KEY; 





(3) 执行 下 面 的 命令 。 

















INSERT INTO tb rank 
(empid,sales,month) 
(SELECT 
empid,sales,month 
FROM tb 
ORDER BY sales DESC); 








我 们 一 个 一 个 地 来 看 上 面 的 操作 方法 。 上 述 操作 方法 中 的 GD ~ @ 分 别 进行 了 如 下 处 理 。 











人 仅 复 制 表 tb 的 列 结构 ， 创 建 表 tb_rank (一 7.3 节 ) 

@) 向 表 tb_rank 中 添加 具有 自动 连续 编号 功能 的 列 c_rank (一 6.3 节 、6.8 节 ) 

(3) 使 用 子 查询 ， 按 sales 由 高 到 低 的 顺序 对 表 tb 进行 排序 ， 并 将 列 empid、 列 sales 和 列 
month 的 记录 插入 到 表 tb_rank 中 








在 步 又 @ 中 , 列 c_rank 中 自动 输入 了 连续 编号 ， 这 个 连续 编号 就 是 排名 。 





























执行 结果 

mysql> SELECT * FROM tb rank; 

3 于 和 3 > + 
empid sales month c_ rank 

+------- +------- +------- +-------- 十 
AlL01 300 5 工 
Al102 205 6 至 
Al01 184 4 2 
A1l04 181 4 4 
A1l03 小 息 4 5 
Al104 93 S 6 
Al107 87 6 7 
A1l02 54 . 8 
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5 | 
| A1l03 | 12 | 6 | 
种 二 号 这 总 写 汪 避 祝 旺 下 全 Fs + 

) 


10 rows in set (0.00 sec 





10.6 ”总结 


本 章 介 绍 了 以 下 内 容 。 


@ 合 并 显示 多 个 表 的 方法 

@ 通 过 内 连接 显示 多 个 表 的 方法 

@ 内 连接 和 外 连接 的 区 别 

@ 自 连接 的 方法 和 排序 的 方法 

@ 通 过 子 查询 实现 两 个 阶段 的 提取 








在 实际 使 用 的 数据 库 中 ,信息 大 多 分 布 在 多 个 表 内 。 如 何 将 多 个 表 连 接 在 一 起 ， 就 得 发 挥 个 人 
的 技术 能 力 了 。 





> 自我 检查 
下 面 检查 一 下 本 章 学 习 的 内 容 是 否 全 部 理解 并 掌握 了 。 




















口 能 够 使 用 UNION 对 多 个 SELECT 的 记录 进行 合 

口 能 够 指定 连接 键 进行 内 连接 ， 从 多 个 表 中 sELECT 

口 能 够 使 用 左 外 连接 和 右 外 连接 ， 从 多 个 表 中 SELECT 必要 的 数据 
口 能 够 理解 子 查询 的 含义 

口 能 够 以 子 查询 获得 的 值 为 条 件 ， 进 一 步 提 取 记 录 


i 练习 题 





问题 1 






































在 表 tb 中 ， 按 照 sales 由 低 到 高 的 顺序 排名 ， 并 从 第 一 名 开始 显示 员工 号 、 销 售 额 和 排名 。 









































HINT 
参考 10.4.2 节 中 介绍 的 排序 方法 。 按照 “由 低 到 高 ”的 顺序 排名 。 
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问题 2 
针对 下 面 的 表 tb1 和 表 tb， 只 从 表 tb 中 提取 表 tb1 中 存在 的 记录 。 




































































































































































戎 表 tb 

A103 101 4 A101 佐 茧 40 只 提取 该 
A102 54 5 A102 高 桥 28 re 和 
A104 181 4 A103 中 川 20 

A101 184 4 A104 渡 边 23 

A103 7 5 A105 西 泽 35 

A101 300 5 

A102 205 6 

A104 93 5 @ 一 一 从 该 表 中 提 

A103 虽 6 RE 

A107 87 6 























HINT 
想 一 下 EXISTS 的 使 用 方法 。 





ta 参考 答案 





问题 1 











执行 下 面 的 命令 。 


SELECT a.empid,a.sales,COUNT (*) 
FROM tb AS a 


JOIN tb AS b 
WHERE a.sales>=b.sales 
GROUP BY a.sales; 





mysql> SELECT a.empid,a.sales,COUNT (*) 
-> FROM tb AS a 
-> JOIN tb AS b 
-> WHERE a.sales>=b .sales 
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-> GROUP BY a.sales; 
+------- +------- +---------- 十 
empid sales COUNT (*) 
+------- +------- +---------- 十 
Al103 下 全 
Al103 17 2 
A1l02 54 3 
A1l07 87 4 
Al04 93 5 
A1l03 0 6 
A1l04 下 有 工 7 
A1l01 184 8 
A1l02 205 EE 
A1l01 300 10 
+------- +------- +---------- 十 
10 rows in set (0.05 sec) 
问题 2 
执行 下 面 的 命令 。 


SELECT * 
FROM tb 


WHERE EXISTS 
(SELECT * FROM tbl1 WHERE tb.empid=tbl .empid); 




















mysql> SELECT * 
-> FROM tb 
-> WHERE EXISTS 
-> (SELECT * FROM tbl WHERE tb.empid=tbl .empid); 
+------- +------- +------- 十 
empid sales month 
+------- +------- +------- 十 
A103 101 4 
A1l02 54 5 
Al04 181 4 
Al01 184 4 
A103 17 5 
Al01 300 全 
A1L02 205 6 
A1L04 33 S 
Al03 小 六 6 
+------- +------- +------- 十 
9 rows in set (0.05 sec) 











另外 ， 因 为 表 tbl 中 不 存在 “A107”， 所 以 没有 显示 “A107”。 
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视图 是 一 种 非常 方便 的 功能 。 该 功能 可 以 隐藏 一 些 重要 的 数 
据 ， 按 照 我 们 想 要 的 条 件 收集 需要 的 记录 。 









































11.1 什么 是 视图 


C 11.1.1 视图 的 真面目 

那么 ， 视 图 到 底 是 什么 呢 ? 前 面 我 们 学 习 了 多 种 使 用 SELECT 提取 记录 的 方法 。 将 SELECT 的 
结果 像 表 一 样 保留 下 来 的 虚 表 就 是 视图 。 

视图 不 是 表 。 因 此 ， 视 图 中 并 没有 保存 记录 或 者 列 中 的 数据 。 也 就 是 说 ， 视 图 是 一 种 信息 ， 用 
于 查询 记录 ， 比 如 在 x x 条 件 下 收集 x x 表 的 x x 列 和 x x。 

















11.1.2 视图 的 用 途 
视图 虽然 看 起 来 像 表 ,但 它 没有 实体 ， 只 是 一 种 信息 。 视 图 的 便利 之 处 在 于 ， 用 户 可 以 按照 想 
要 的 条 件 收集 某 表 中 某 列 的 数据 ( 见 图 11-1 )。 
之 前 那 种 通过 设置 条 件 从 表 中 提取 记录 的 做 法 非常 麻烦 。 如 果 相 同 的 提取 操作 以 视图 的 方式 执 
行 ， 就 可 以 将 视图 作为 符合 用 户 个 人 喜好 的 表 来 使 用 。 
从 用 户 的 角度 来 看 ， 视 图 和 表 在 使 用 上 并 没有 什么 区 别 。 和 表 一 样 ， 视 图 也 可 以 进行 SELECT 
和 UPDATE。 如 果 更 新 视图 的 记录 ， 基 表 的 记录 也 会 更 新 。 
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需要 的 列 
需要 的 列 
实际 的 表 3 
实际 的 表 2 


需要 的 列 编号 ， 津贴 1 ， 津贴 2 ， 津 贴 3 ， 
实际 的 表 1 编号 ， 销 售 额 1 销售 额 2! 销售 额 3! … 1 1 1 


编号 ;， 姓名 ， 邮 编 ， 住 址 ， 






















































































条 件 3 





通过 必要 的 条 件 
选取 需要 的 列 





编号 ， 姓名 ， 销 售 额 2! 销售 额 3， 津贴 3 ; 


























11-1 视图 

















另外 ， 对 于 一 些 不 能 被 修改 的 重要 数据 ， 我 们 可 以 只 让 管理 员 等 具有 特殊 权限 的 人 来 操作 相关 
的 表 ， 同 时 再 准备 一 个 收集 了 无 关 紧 要 部 分 的 视图 ， 这 样 就 很 安全 了 。 

此 外 ， 数 据 库 高 级 用 户 可 以 为 初学 者 创建 容易 理解 的 视图 来 代替 不 易 理 解 的 表 ， 这 也 是 视图 的 
用 途 之 一 。 











[Uy 








视图 和 MySQL 的 版 本 
与 安装 MAMP 的 方法 无 关 ， 实 际 上 视图 只 能 在 MySQL 5 或 更 高 的 版 本 中 使 用 。 工 作 中 我 们 会 用 到 各 种 版 


本 的 MySOL， 因 此 建议 在 编写 SQL 语句 之 前 检查 一 下 MySQL 的 版 本 。 































































































MySQL 监视 器 会 在 启动 的 时 候 显示 版 本 。 另 外 ， 我 们 也 可 以 使 用 8.2.3 节 中 介绍 的 VERSION 函数 来 查看 


版 本 。 











c:NXUsers>mysdl -u root -proot 

Warning: Using a password on the command line interface can be insecure. 
Welcome to the MySQL monitor. Commands end with ; or \g. 

Your MySQL connection ia is 15 

Server version: 5.6.34-1og MySQL Community Server (GPL) 


ersh rie Toe 2000RE20T5 oe cue) oe cu ler ele 


eSemveds a 
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11.2 ”使 用 视图 


11.2.1 创建 视图 


< 和 QR 


我 们 来 试 着 创建 视图 吧 。 虽 说 是 “创建 "， 但 因为 视图 没有 实体 ， 所 以 说 成 “定义 视图 ”可 能 
更 为 准确 。 


创建 视图 














上 面 的 语句 表示 “对 SELECT 的 记录 CREATE VIEW ( 视图。 不 要 忘记 在 SELECT 的 前 钙 
加 As。 














上 述 语 句 中 的 “WHERE 条 件 ”， 实 际 上 也 可 以 换 成 ORDER BY、LIMIT 和 JOIN 等 。 总 之 ， 
要 在 某 些 条 件 下 选择 列 ， 创 建 虚拟 的 表 。 也 就 是 将 前 面 执行 的 许多 SELECT 结果 通过 CREATE 
VIEW ..，RAS 创建 为 视图 。 

下 面 我 们 试 着 创建 一 个 简单 的 视图 。 员 工 信 息 表 tbl 由 员工 号 (empid )、 姓 名 (name ) 和 年 龄 
(age ) 组 成 。 试 着 创建 没有 员工 号 、 只 包含 姓名 和 年 龄 这 两 个 列 的 视图 v1。 

为 了 避免 更 新 表 tbl 中 的 内 容 ， 这 里 我 们 使 用 与 员工 信息 表 tbl 内 容 相同 的 表 tb1J。 具 体 来 说 ， 
就 是 创建 由 表 tp1J 的 姓名 (name ) 与 年 龄 (age ) 两 个 列 构成 的 视图 v1， 并 显示 视图 v1 的 内 容 。 























PP 执行 内 容 


bb 生前 [天 Wb > 执行 后 ( 创建 视图 v1 ) 

































































empid name age name age 
A101 佐 蔷 40 佐 茧 40 
A102 “| 高 桥 28 中 高 桥 ”| 28 
A103 中 川 20 中 川 20 
A104 渡 边 23 渡 边 23 
A105 上 泽 35 西 泽 35 









































只 包含 这 两 列 
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操作 方法 


中 执行 下 面 的 命令 。 

: CREATE VIEW v1 

: AS 

: SELECT name,age 
: FROM tb1J; 




















mysql> SELECT * FROM V1; 



































瑟 且 二 二 二 过 有 + 
name age 
ns ES 十 
佐 茧 40 
高 村 28 
中 川 20 
渡 边 23 
6 泽 35 
a es 十 


5 rows in set (0.03 sec) 














使 用 SELECT 显示 视图 记录 的 方法 ， 和 以 表 为 操作 对 象 时 使 用 的 方法 完全 相同 。 这 次 我 们 只 从 
一 个 表 中 提取 了 列 ， 实 际 上 也 可 以 使 用 多 个 表 ， 通过 设置 条 件 来 提取 想 要 的 内 容 。 














视图 只 显示 了 基 表 的 一 部 分 。 因 此 ， 如 果 更 新 了 基 表 的 值 ， 收 集 并 显示 基 表 值 的 视图 的 值 也 会 
更 新 。 

那么 ， 如 果 更 新 了 视图 的 值 ， 基 表 的 值 又 会 怎样 呢 ?” 实 际 上 ， 视 图 不 仅 是 基 表 的 一 部 分 ， 它 也 
是 指向 基 表 数据 的 窗口 。 因 此 ， 如 果 更 新 视图 的 值 ， 基 表 的 值 也 会 随 之 更 新 。 

我 们 来 试 着 更 新 一 下 视图 v1 的 记录 ， 看 看 视图 的 记录 更 新 是 否 反 映 到 了 基 表 中 。 将 视图 vl 中 
A101“ 伍 且 ” 的 姓名 更 新 为 “主任 . 佐藤”。 与 更 新 表 时 使 用 的 方法 一 样 ， 我 们 需要 使 用 UPDATE .…. 
SET 来 更 新 视图 的 值 。 




































































> 执行 前 ( 视图 v1 ) 




























































































> 执行 后 
name age 
主任 佐 茧 40 
高 桥 28 
中 川 20 
渡 边 23 
西 泽 35 



































name age 
把 该 记录 更 
新 为 “主任 。 一 9 佐藤 40 
佐藤 ” 高 桥 28 
中 川 20 
渡 边 23 
西 泽 35 
操作 方法 




















首先 ， 我 们 来 确认 一 下 视图 v1 是 否 被 正确 更 新 了 。 
请 执行 下 面 的 SELECT 语句 。 


SELECT * FROM v1; 
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mysql> SELECT * FROM vi1; 
+------------ +------ 
name age 
+------------ 二 -一 -一 -一 
主任 . 估 芯 40 
高 桥 28 
中 川 20 
渡 边 呈 3 
西 泽 35 
+------------ 二 -一 -一 一 一 
5 rows in set (0.00 








视图 v1 已 经 被 顺利 更 新 了 。 那 么 ,视图 v1 的 基 表 tb1J 又 怎样 呢 ? 


SELECT * FROM tbl1yJ; 
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mysql> SELECT * FROM tbl1yJ; 
= 六 二 宇和 刷 写 三 + 
empid name age 
站 下 让 + 
Al01 主任 .佐藤 40 
ARA102 高 村 28 
RA103 中 川 20 
A104 渡 边 23 
RA105 泽 35 
4 Fo 4 + 
5 rows in set (0.00 sec) 


























看 来 基 表 tb1J 的 值 也 更 新 了 。 由 此 我 们 可 以 得 知 ， 如 果 更 新 视图 的 值 ， 其 基 表 的 记录 也 会 随 之 
更 新 。 








11.3 ”设置 条 件 创建 视图 


e 11.3.1 设置 条 件 创建 视图 

在 11.2.1 节 中 ,我 们 只 从 一 个 表 中 收集 了 任意 的 两 个 列 来 创建 视图 。 这 次 ， 我们 将 从 两 个 表 中 
提取 记录 并 通过 WHERE 设置 条 件 来 创建 视图 。 

首先 ， 在 销售 信息 表 tb 中 提取 销售 额 大 于 等 于 100 万 元 的 优秀 记录 ， 然 后 连接 员工 信息 表 tb1J 
显示 该 员工 的 姓名 。 内 连接 JOIN 的 使 用 方法 请 参考 10.2.1 节 。 

连接 表 tb1J 与 表 tb， 提 取 销 售 额 (sales ) 大 于 等 于 100 万 元 的 记录 ， 创 建 由 表 了 b 的 员工 号 
(Cempid ) 和 销售 额 (sales )， 以 及 表 tb1J 的 姓名 (name ) 构成 的 视图 V2， 然后 显示 视图 v2 的 所 有 
记录 。 
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P 执行 内 容 




















> 执行 前 ( 表 tb ) > 执行 前 ( 连接 表 tb1 ) 
empid sales month 使 用 empid 连 接 一 ®@| empid name age 

















: 



































































































































A102 54 5 
提取 sales 大 于 等 于 100 的 记录 
4 
A103 17 5 中 
5 
6 
A104 93 5 
A103 12 6 
A107 87 6 
> 执行 后 ( 视图 v2 ) 
empid | name sales 
A103 中 川 101 
A104 “| 渡 边 181 
A101 “| 主任. 佐 蕨 “| 184 
A101 | 主任. 佐 芯 ”| 300 
A102 “| 高 桥 205 
操作 方法 
Gd) 执行 下 面 的 命令 。 
CREATE VIEW v2 
AS 
SELECT tb.empid,tblJ.name,tb.sales 
FROM tb 
JOIN tb1lJ 
USING (empid) 


WHERE tb.sales>=100; 


@ 执行 下 面 的 命令 。 


SELECT * FROM V2; 
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执行 结果 
mysql> CREATE VIEW v2 
-> AS 
-> SELECT tb.empid,tblJ.name,tb.sales 
-> FROM tb 
-> JOIN tblJ 
-> USING (emPid) 


-> WHERE tb.sales>=100; 
Query OK, 0 rows affected (0.01 sec) 


mysql> SELECT * FROM V2; 




















> i | + 
empid name sales 
FS = 和 于 富生 + 
ARA103 中 川 0 
ARA104 渡 边 181 
Al01 主任 .佐藤 184 
A101 主任 .佐藤 300 
RA102 高 村 205 
束 呈 外 全 全 全 已 所 下 已 属 记 全 可 已 必 全 本 过 后 十 


5 rows in set (0.19 sec) 

















视图 v2 被 创建 了 出 来 ,并且 显示 了 sales 值 大 于 等 于 100 的 5 条 记录 。 











前 一 节 我 们 创建 了 视图 v2， 提 取 了 sales 大 于 等 于 100 的 记录 。 表 tb 中 从 上 往 下 数 排 在 第 2 位 
的 员工 “A102” 的 销售 额 小 于 100， 所 以 不 会 显示 在 视图 v2 中 。 









































> 表 tb 

empid sales month 
A103 101 4 
A102 54 8 
A104 181 4 
A101 184 4 
A103 17 5 
A101 300 5 
A102 205 6 
A104 93 5 
A103 12 6 
A107 87 6 
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那么 ， 如 果 将 这 个 值 修改 成 大 于 等 于 100 的 数字 ， 也 就 是 在 创建 好 视图 后 满足 了 提取 条 件 ， 视 
图 会 发 生 什么 变化 呢 ? 

我 们 来 实际 操作 一 下 。 对 于 表 了 世 中 empid 为 “A102”、sales 为 “54” 的 记录 ,将 sales 的 值 修 
改 为 “777”， 然 后 显示 11.3.1 节 中 创建 的 视图 v2。 














PP 执行 内 容 

































































































































































> 执行 前 ( 表 tb ) > 执行 后 
empid sales month empid sales month 
A103 101 4 A103 101 4 将 "54” 修 改 为 
A102 |54 5 M77 Ei a ， 0 
A104 181 4 A104 181 4 条 件 
A101 184 4 »》 A101 184 4 
A103 17 5 A103 17 5 
A101 300 5 A101 300 5 
A102 205 6 A102 205 6 
A104 93 5 A104 93 5 
A103 12 6 A103 12 6 
A107 87 6 A107 87 6 
D4 
> 执行 后 ( 视图 v2 ) 
empid name sales 
A103 中 川 101 
A102 “| 高 桥 > 
A104 渡 边 181 








A101 主任 . 佐 芯 | 184 
A101 主任 佐 芯 ”| 300 
A102 高 桥 205 






































操作 方法 





() 执行 下 面 的 命令 。 

















UPDATE tb SET sales=777 WHERE sales=54; 





236 | 第 11 章 熟练 使 用 视图 


@@ 执行 下 面 的 命令 。 








mysql> UPDATE tb SET sales=777 WHERE sales=54; 
Query OK, 1 row affected (0.01 sec) 
Rows matched: 1 Changed: 1 Warnings: 0 


mysql> SELECT * FROM v2; 




















下 Ns 下 十 
empid name sales 
+------- +----------- +------- + 
A1l03 中 川 于 0 
RA102 高 桥 了 7 
A104 渡 边 181 
A1l01 主任 * 佐 匡 184 
A1l01 主任 .佐藤 300 
RA102 高 村 205 
Sones ne te 十 


6 rows in set (0.00 sec) 











可 以 看 到 视图 v2 的 值 也 更 新 了 。 也 就 是 说 ， 在 符合 视图 设置 条 件 的 情况 下 ， 如 果 基 表 更 新 ， 
视图 中 的 记录 也 会 随 之 更 新 。 设 置 了 条 件 的 视图 始终 会 显示 与 条 件 相 匹配 的 记录 。 
sales = 777 看 起 来 有 些 奇 怪 ， 所 以 请 执行 下 面 的 语句 将 值 恢复 成 原来 的 “54”。 


UPDATE tb SET sales=54 WHERE sales=777; 


C 11.3.3 ”确认 视图 

下 面 我 们 来 学 习 一 下 确认 视图 是 否 存 在 的 方法 和 确认 视图 结构 的 方法 。 表 和 视图 的 处 理 基本 上 
是 相同 的 。 

我 们 需要 通过 “sHOW TABLES;” 来 确认 存在 哪些 视图 。 


确认 视图 
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视图 会 与 表 交 织 在 一 起 显示 出 来 。 


mysql> SHOW TABLES; 














XX rows in set (0.03 sec) 








和 表 一 样 ， 列 结构 可 以 通过 DESC 显示 出 来 。 
本 吕 未 视 轩 的 列 人 寺 构 














| 执行 结果 























mysql> DESC v2; 

+------- +------------- +------ +----- +--------- +------- 十 
| Field | Type | Null | Key | Default | Extra | 
+------- +------------- +------ +----- +--------- +------- 十 
| empiqd | varchar(20) | YES | | NUL | 

| name | varchar(10) | YES | | NUL | 

| sales | int(11) | YES | | NUL | | 
+------- +------------- +------ +----- +--------- +------- + 
3 rows in set (0.33 sec) 








与 表 相同 ， 视 图 的 详细 信息 也 可 以 通过 以 下 方法 显示 (一 13.2.1 节 )。 





显示 视图 的 详细 信息 
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11.4 ”限制 通过 视图 写 入 


11.4.1 对 视图 执行 INSERT 操作 会 出 现 什么 样 的 结果 


OO 


在 11.2.2 节 中 ， 我 们 通过 视图 更 新 了 基 表 的 记录 。 这 次 ， 我 们 试 着 通过 视图 来 INSERT 记录 。 
视图 没有 实体 ， 它 只 是 一 个 虚拟 表 。 如 果 将 记录 插入 到 这 个 虚拟 视图 中 ， 会 出 现 什 么 样 的 结果 呢 ? 

视图 是 用 户 从 基 表 上 随意 收集 的 列 。 换 句 话 说， 作为 视图 可 见 的 部 分 大 多 是 基 表 中 的 一 部 分 内 
容 。 对 视图 执行 INSERT 操作 ， 就 意味 着 只 能 向 表 的 其 中 一 部 分 内 容 中 插入 数据 。 
前 面 介 绍 得 有 点 嘿 唆 ， 用 一 句 话 概括 就 是 对 视图 执行 INSERT 操作 是 有 限制 的 。 例 如 ， 在 使 用 
了 UNION、JOIN、 子 查询 的 视图 中 ， 不 能 执行 INSERT 和 UPDATE。 但 如 果 像 11.2.1 节 中 创建 的 
视图 一 样 ， 只 是 从 一 个 表 中 提取 了 列 ， 那 么 执行 INSERT 和 UPDATE 是 没有 任何 问题 的 。 

在 11.2.1 节 中 创建 的 视图 vl 仅 包含 两 个 列 ， 即 表 tb1] 的 列 name 和 列 age。 试 着 向 列 name 和 
列 age 中 插入 数据 。 


插入 方法 和 向 表 中 插入 数据 的 方法 相同 。 试 着 向 视图 vl 中 插入 name=' 临时 工 . 石 
age=18 的 记录 。 







































































用 





> 执行 内 容 























































































































> 执行 前 ( 视图 v1 ) > 执行 后 

name age name age 
主任 . 佐 茧 “| 40 主任 ' 佐 芒 。 | 40 
高 桥 28 高 桥 28 
中 川 20 中 川 20 
渡 边 23 渡 边 23 

泽 35 西 泽 35 se 
EE 


























操作 方法 
执行 下 面 的 命令 。 
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Query OK, 1 row affected (0.03 sec) 





mysql> INSERT INTO v1 VALUES(' 临 时 工 : 石 田 ' ,18); 








首先 我 们 来 确认 一 下 插入 数据 后 视图 v1 的 内 容 。 


SELECT * FROM v1; 


















































mysql> SELECT * FROM v1; 
二 生生 和 +------ + 
name age 
+------------- +------ 十 
主任 -佐藤 40 
高 桥 28 
中 川 20 
渡 边 3 
泽 35 
年 时 工 "石田 18 
+------------- +------ 十 
6 rows in set (0.00 sec) 














姓名 和 年 龄 被 完整 地 插入 到 了 视图 v1 中 。 那 么 ， 





视图 vl 的 基 表 tb1J 会 变 成 什么 样 呢 ? 


SELECT * FROM tb1J; 


















































mysql> SELECT * FROM tb1J; 

要 守 六 入 td i 后 
empid name age 
于 A ee i + 
A1l01 主任 :佐藤 40 
A102 高 桥 28 
A1l03 中 川 20 
RA104 渡 边 23 
RA105 西 泽 35 
NULL 备 时 工 * 石 田 18 
种 各 主 训 包 和 和 + 

6 rows in set (0.00 sec) 

















看 来 “临时 工 : 石田 ”和 “18” 这 两 个 数据 都 TNSERT 成 功 了 。 男 外 ， 对 于 在 视图 v1 中 没有 
被 定义 的 列 empid， 因 为 没有 插入 数据 ， 所 以 输入 了 NULL。 
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11.4.2 设置 了 条 件 的 基 表 中 会 发 生 什么 变 


如 果 在 设置 了 条 件 的 视图 中 插入 违反 条 件 的 数据 ， 基 表 会 发 生 什么 变化 呢 ? 我 们 来 试 试看 吧 。 

首先 ， 以 “销售 额 大 于 等 于 100 万 元 ”为 条 件 ， 创 建 仅 包括 销售 信息 表 tb 的 列 empid 和 列 
sales 的 基础 视图 。 

具体 来 说 ， 就 是 以 sales>=100 为 条 件 ， 创 建 仅 包括 表 tb 的 列 empid 和 列 sales 的 视图 v3， 
然后 显示 视图 v3 的 所 有 记录 。 


P 执行 内 容 



































































































































> 执行 前 ( 表 tb ) 区 执行 后 ( 视图 v3 ) 
A103 101 4 A103 Wo 
中 建 含 列 | 
| 
A101 184 4 A101 300 
A103 17 5 A102 205 
A101 300 5 
A102 205 6 
A104 93 5 
A103 12 6 
A107 87 6 
提取 列 值 大 于 等 于 100 的 记录 




















操作 方法 


( 执行 下 面 的 命令 。 

: CREATE VIEW v3 

: RS 

SELECT empid,sales 
FROM tb 

WHERE sales>=100; 




















11.4 限制 通过 视图 写 入 


| 所 





mysql> CREATE VIEW v3 


-> AS 
-> SELECT empid,sales 
-> FROM tb 


-> WHERE sales>=100; 
Query OK, 0 rows affected (0.01 sec) 


mysql> SELECT * FROM v3; 


es Re + 
| empid | sales | 
二 头号 
| Alo3 | 101 | 
| Alo4 | 181 | 
| Alol | 184 | 
| Alol | 300 | 
| Alo2 | 205 | 
人 bm tt 机 





5 rows in set (0.06 sec) 








可 以 看 到 ， 视 图 v3 以 销售 额 大 于 等 于 100 万 元 的 记录 为 对 象 ， 显 示 了 列 empid 和 列 sales。 这 








里 再 重复 一 遍 ， 视 图 v3 仅 包含 sales 大 于 等 于 100 的 记录 。 





> 插入 不 符合 视图 条 件 的 记录 


这 里 ， 我 们 向 无 法 显示 销售 额 小 于 100 万 元 的 记录 的 视图 v3 中 搬入 小 于 100 的 值 。 那 么 ， 视 


图 v3 会 出 现 什 么 样 的 情况 呢 ? 下 面 的 命令 用 于 向 列 sales 中 INSERT 数值 “50”。 


INSERT INTO v3 VALUES (' 有 恶意 刁难 ! ,50)， 








mysql> INSERT INTO v3 VALUES (' 恶意 刁难 ' ,50); 
Query OK, 1 row affected (0.08 sec) 











轻松 输入 进去 了 。 那 么 ,视图 v3 会 如 何 显示 呢 ? 因为 小 于 100 的 值 无 法 显示 ， 所 以 执行 





“SELECT * FROM v3;” 应 该 看 不 到 刚才 插入 的 值 。 





mysql> SELECT * FROM v3; 
+------- +------- 十 
| empiqd | sales | 
+------- +------- 十 
| Al103 
| Al104 
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| A1l01 | 184 | 
| Al01 | 300 | 
| A102 | 205 | 
a 人 + 


5 rows in set (0.02 sec) 





那么 基 表 tb 会 变 成 什么 样 呢 ? 


SELECT empid,sales FROM tb; 

















执行 结果 
mysql> SELECT empid,sales FROM tb; 
+---------- +------- + 
empid sales 
+---------- +------- + 
A103 101 
A1l02 777 
A1l04 181 
A1l01 184 
A1l03 二 次 
A1L01 300 
A102 205 
Al04 93 
A1l03 12 
A1l07 87 
恶意 刁难 50 
二 = +------- 十 
11 rows in set (0.00 sec) 

















基 表 tb 中 竟然 插入 了 这 个 值 ! 在 默认 情况 下 ，WHERE 条 件 会 被 忽略 ， 数 据 会 INSERT 到 基 表 中 。 
在 某 些 情况 下 ,没有 插入 数据 的 列 中 会 输入 NULL。 另 外 ， 表 tb 中 插入 的 “恶意 刁难 ”的 记录 
会 在 以 后 带 来 麻烦 ， 所 以 请 参考 9.3.2 节 的 内 容 进 行 删 除 。 


MN 


当 通 过 视图 INSERT 记录 时 ， 即 使 与 WHERE 的 条 件 不 匹配 ， 数 据 也 会 直接 输入 到 基 表 中 。 这 
是 否 就 意味 着 总 是 可 以 输入 与 WHERE 条 件 不 符 的 记录 呢 ? 

但 是 ， 对 于 有 条 件 限制 的 视图 ， 无 视 条 件 输入 记录 有 时 会 带 来 一 定 的 麻烦 。 男 外 ， 从 视图 
入 的 记录 无 法 在 该 视图 中 确认 也 是 一 件 很 麻烦 的 事 。 

为 了 应 对 这 些 情况 ， 我 们 可 以 将 视图 设置 成 “不 接受 与 条 件 不 匹配 的 记录 ”。 为 了 防止 输入 与 
WHERE 条 件 不 匹配 的 记录 ， 我 们 可 以 在 使 用 CREATE VIEW 创建 视图 时 ， 加 上 WITH CHECK 
OPTION。 














P 输 
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下 面 是 加 上 WITH CHECK OPTION 来 创建 视图 v4 的 示例 。 


CREATE VIEW v4 
AS 
SELECT empid,sales 


FROM tb 
WHERE sales>100 
WIIH CHECK OPTION; 








像 上 面 这 样 加 上 WITH CHECK OPTION 后 ， 如 果 执 行 














INSERT INTO v4 VALUES(' 





试图 插入 不 符合 条 件 的 记录 ， 就 会 发 生 错误 。 


mysql> INSERT INTO v4 VALUES (' 恶意 刁难 ' ,50); 
ERROR 1369 (HY000): CHECK OPTION failed 'dbl.v4' 














这 样 就 无 法 输入 不 符合 条 件 的 记录 了 。 


11.5 替换、 修改 和 删除 视图 


下 面 介 绍 编辑 视图 的 方法 。 

















S 11.5.1 替换 视图 

当 存 在 同名 的 视图 时 ， 如 何 对 其 进行 替换 呢 ? 下 面 就 来 介绍 具体 的 方法 。 

当 执 行 CREATE VIEW 时 ， 如 果 已 经 有 同名 的 视图 存在 ， 该 命令 会 报错 。 在 这 种 情况 下 ， 可 以 
像 “CRERATE OR REPLACE VIEW .…” 这 样 ， 加 上 OR REPLACE 替换 视 网 。 也 就 是 说 ， 删 除 已 
经 存在 的 同名 的 视图 ， 创 建新 的 视图 。 

例如 ， 在 视图 vl 已 经 存在 的 情况 下 执行 以 下 命令 。 
































CREATE OR REPLACE VIEW VI 


AS 
SELECT NOW(); 
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mysql> CREATE VIEW v1 
-> AS 
-> SELECT NOW(); 
ERROR 1050 (42S01): Table 'v1l' already exists 
mysql> CREATE OR REPLACE VIEW v1 
-> AS 
-> SELECT NOW(); 
Query OK, 0 rows affected (0.09 sec) 


mysql> SELECT * FROM v1; 


全 二 时 人 全 二 生 二 十 
| NOW() | 
上 + 
| 2018-07-02 10:21:35 | 
es 十 





1 row in set (0.05 sec) 








视图 应 该 被 顺利 替换 了 。 视 图 v1 被 删除 ， 并 替换 为 始终 显示 当前 日 期 和 时 间 的 视图 “v1”。 
(下 一 节 会 对 视图 v1 进行 恢复 。 ) 

这 是 在 不 知道 是 否 有 同名 视图 存在 的 状态 下 创建 视图 的 方法 。 在 批量 执行 SQL 语句 (一 14.2.1 
节 ) 的 情况 下 ， 不 管 同名 的 视图 是 否 存 在 ， 该 命令 都 能 够 创建 视图 ， 非 常 方 便 。 








S 11.5.2 ”修改 视图 结构 


当 修 改 视 图 结构 时 ， 我 们 需要 使 用 ALTER VIEW。 
医 到 本 修改 视图 结构 




















与 使 用 “CREATE VIEW ... AS SELECT” 创 建 视图 的 方法 基本 相同 。 在 上 一 节 中 ， 我 们 让 视 
图 vl 仅 显 示 NOW () 的 内 容 ， 这 里 来 把 视图 v1 恢复 成 原来 的 样子 。 
通过 下 面 的 操作 ， 可 以 让 当前 的 视图 vl 中 包含 表 tbl 的 列 name 和 列 age。 





ALTER VIEW v1 
AS 


SELECT name,age 
FROM tbl; 
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| 执行 结果 


mysql> ALTER VIEW v1 





-> AS 
-> SELECT name,age 
= FROM tbl; 


Query OK, 0 rows affected (0.33 sec) 


mysql> SELECT * FROM v1; 























+------ +------ 十 
| name | age | 
+------ +------ 十 
| 佐藤 | 40 | 
| 高 桥 | 28 | 
| 中 川 | 20 | 
| 渡 边 | 23 | 
| 西 泽 | 35 | 
+------ +------ 十 


5 rows in set (0.08 sec) 











按照 下 面 介 绍 的 方法 先 删 除 视图 ， 然 后 执行 “CREATE VIEW ...”， 也 会 得 到 相同 的 结 





C 11.5.3 删除 视图 


在 MySQL 中 删除 某 个 对 象 时 ， 需 要 使 用 DROP 命令 。 删 除 视图 也 需要 使 用 DROP 命令 。 删 除 
视图 需要 用 到 的 命令 与 删除 数据 库 和 表 时 一 样 。 


删除 视图 


























但 是 ， 如 果 删 除 的 视图 不 存在 ， 该 命令 就 会 报错 。 
如 果 像 下 面 这 样 加 上 IF EXISTS， 那 么 即使 目标 视图 不 存在 也 不 会 报错 ， 只 是 不 执行 删除 操作 而 已 。 


DROP VIEW IF EXISTS v1; 


ED 什么 是 复制 
复制 ( replication ) 这 项 技术 作为 提高 数据 库 处 理 效率 和 进行 备份 的 手段 ， 在 实际 工作 中 不 可 或 缺 。 虽 然 本 
书 不 会 介绍 复制 的 具体 操作 方法 ， 但 是 作为 知识 储备 ， 我 们 最 好 了 解 一 下 复制 的 基本 内 容 。 
















































































全 什么 是 复制 
replica 指 的 是 复制 品 。 历 史 雕 像 和 奖牌 的 仿制 品 、 高 级 手表 的 仿造 品 、 假 币 等 都 是 复制 品 。 复 制 是 指 制 作 
此 类 复制 品 ， 也 就 是 制作 副本 。 
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数据 库 中 的 复制 是 指 创建 原始 数据 库 的 副本 ( 复制 品 )， 通 过 灵活 使 用 原始 数据 库 和 数据 库 副 本 来 有 效 进行 
处 理 的 技术 。 例 如 ， 复 制 原始 数据 库 ， 将 内 容 完全 相同 的 数据 库存 储 在 另 一 台 服 务 器 上 。 因 为 每 个 服务 器 上 的 数 
据 库 都 是 相同 的 ， 所 以 能 够 通过 分 布 式 处 理 来 加 快 整体 的 处 理 速度 。 

原始 数据 库 称 为 主 库 ( master )， 通 过 复制 创建 的 副本 称 为 从 库 ( slave )。 

还 有 一 种 配置 多 个 主 服 务 器 的 “多 主 ” 的 方式 。 虽 然 MySQL 中 也 可 以 配置 “多 主 ” 类 型 的 复制 ， 但 这 里 我 
们 要 介绍 的 是 1 个 主 库 对 应 1 个 或 多 个 从 库 的 “ 主 从 ”类 型 的 复制 。 


















































































































































全 作为 备份 使 用 
从 库 的 内 容 基 本 上 和 主 库 一 样 ， 所 以 可 以 作为 备份 使 用 。 在 主 库 发 生 故障 的 情况 下 ， 如 果 切 换 到 从 库 执行 所 
有 的 处 理 ， 就 能 够 在 短 时 间 内 重新 启动 系统 。 




























































































@ MySQL 复制 的 特征 
通过 复制 来 创建 数据 库 的 副本 可 以 提高 处 理 效率 ， 因 为 可 以 进行 分 布 式 工 作 。 

以 MySQL 为 例 ， 分 布 式 处 理 是 在 主 库 进行 UPDATE 和 INSERT 等 写 入 处 理 ， 在 从 库 进 行 SELECT 等 读 取 
处 理 。 以 这 样 的 方式 进行 处 理 ， 每 个 服务 器 都 能 充分 发 挥 作用 ， 从 而 提高 效率 。 
要 想 构建 复制 的 架构 ， 主 库 和 从 库 的 数据 必须 相同 。MySQL 基本 上 是 通过 把 主 库 的 二 进 和 
使 主 库 和 从 库 的 内 容 相同 的 ( 见 图 11-2 )。 所 谓 二 进 制 志 ， 简 单 来 说 就 是 通过 处 理 的 记录 来 描述 “更 新 x x 的 
表 为 x x ”这 样 的 内 容 。 以 这 个 记录 为 基础 ， 如 果 从 库 能 够 如 实地 重 现 该 处 理 ， 就 可 以 使 主 库 和 从 库 的 内 容 相 同 。 


[| 


分 布 式 处 理 


至 卓志 
使 二 者 相同 


图 11-2 复制 的 架构 


这 里 ， 在 什么 时 间 点 将 主 库 的 数据 复制 到 从 库 是 一 个 问题 。 如 果 在 使 用 从 库 进行 查询 的 时 候 主 库 上 发 生 了 更 
新 ， 那 么 严格 来 说 ， 从 库 上 的 查询 结果 是 不 正确 的 。 
实际 上 ， 复 制 分 同步 模式 和 异步 模式 两 种 类 型 。 在 同步 模式 中 ， 主 库 和 从 库 保 持 着 联系 ， 采取“ 停止 工作 ( 等 
待 ) 直到 彼此 的 内 容 完全 相同 ”的 策略 。 所 以 ， 主 库 和 从 库 的 数据 总 是 相同 的 ， 但 是 整体 处 理 也 会 因此 而 延迟 很 多 。 

一 方面 ， 在 异步 模式 中 ， 主 库 和 从 库 不 会 保持 这 样 的 联系 ， 所 以 主 库 和 从 库 的 内 容 也 不 一 定 相同 。 
MySQL 采用 的 是 异步 模式 ， 二 进 制 日 志 会 单方 面 从 主 库 发 送 到 从 库 。 大 家 要 记 住 在 MySQL 中 “ 主 库 和 从 库 的 
内 容 有 时 可 能 不 同 ” 这 一 点 。 
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全 半 同 步 复制 

MySOQL 的 复制 是 异步 的 ， 但 是 MySOL 5.5 以 及 更 高 的 版 本 可 以 实现 一 部 分 过 程 的 同步 。 虽 然 MySQL 的 异 
步 复 制 可 以 将 主 库 上 的 处 理 重 现在 从 库 上 ， 但 其 实 主 库 只 是 简单 地 把 二 进 制 日 志 传送 给 了 从 库 。 至 于 是 否 正确 地 
传送 给 了 从 库 ， 则 没有 进行 确认 。 
















































































11.5 替换、 修改 和 删除 视图 | 247 



































而 在 半 同 步 复 制 的 情况 下 ， 二 进 制 日 志 传 送 给 从 库 后 ， 从 库 会 向 主 库 发 送 响 应 消息 。 图 11-3 显示 了 半 同 步 
复制 的 过 程 。 























四 响应 





主 库 Cj 二进制 日 志 一 一 一 季 从 库 














名 在 主 库 中 数据 写 入 完成 ( COMMIT ) @ 团 在 从 库 中 写 入 数据 





11-3 半 同 步 复制 


为 了 向 主 库 发 送 响应 消息 ， 在 主 库 发 送信 息 到 从 库 之 前 主 库 和 从 库 需 要 同步 。 但 是 ， 从 库 中 数据 的 写 入 操作 
是 异步 的 ， 因 此 称 为 半 同 步 。 半 同步 复制 与 异步 复制 相 比 ， 数 据 可 靠 性 较 强 ， 但 是 整体 处 理 时 间 也 变 长 了 。 




















ec 
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全 无 损 半 同步 复 币 














































































































MySQL 5.7 中 增加 了 许多 与 复制 相关 的 功能 ， 其 中 之 一 就 是 无 损 半 同 步 复 制 。 
上 面 介绍 的 半 同 步 复制 是 将 数据 完全 写 入 主 库 并 发 送 二 进 制 日 志 ， 然 后 从 库 进行 响应 。 而 从 版 本 5.7 开始 
MySQL 使 用 的 无 损 半 同步 复制 则 是 发 送 二 进 制 日 志 ， 收 到 从 库 的 响应 后 在 主 库 上 完成 数据 的 写 入 ( 见 图 11-4 )。 




























































































人 @@) 响应 
@) 发 送 一 一 
主 库 cj 二进制 日 志 一 jy> 从 库 
@ 在 主 库 中 数据 写 入 完成 ( COMMIT ) @) 在 从 库 中 写 入 数据 





图 11-4 无 损 半 同步 复制 
主 库 在 收 到 从 库 的 响应 后 才 写 入 数据 ， 这 就 保证 了 主 库 中 写 入 的 信息 一 定 发 送 给 了 从 库 。 
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本 章 介 绍 了 以 下 内 容 。 


@ 视 图 与 表 不 同 ， 它 没有 实体 
@ 视 图 的 优点 
@ 创 建 视 图 的 方法 


@ 如 何 对 设置 了 条 件 的 视图 进行 更 新 ， 以 及 该 操作 对 基 表 的 影响 


使 用 CREATE TABLE 创建 的 表 称 为 实 表 。 实 表 是 将 数据 实际 写 人 硬盘 等 存储 设备 的 表 。 虽 然 
视图 可 以 像 表 一 样 使 用 ,但 它 没有 实体 。 数 据 库 管理 是 不 允许 出 现 错误 的 。 从 保护 基 表 的 意义 上 来 
说 ， 视 图 的 灵活 使 用 也 是 一 项 重要 的 技术 。 





> 自我 检查 











下 面 检查 一 下 本 章 学 习 的 内 容 是 否 全 部 理解 并 掌握 了 。 











口 能 够 理解 视图 到 底 是 什么 


口 能 够 使 用 CREATE VIEW 创建 视图 
口 能够 创建 设置 了 条 件 的 视图 






























































































































































口 掌握 允许 插入 不 符合 条 件 的 记录 的 方法 ， 以 及 禁止 插入 这 类 记录 的 方法 
二 练习 题 
问题 1 
针对 下 面 的 表 tb， 创 建 视 图 v_sales。 视 图 v_sales 以 列 sales 大 于 等 于 50 的 记录 为 对 象 ， 按 照 
empid 分 组 ， 并 降序 显示 各 组 sales 平均 值 大 于 等 于 120 的 记录 。 
PP ( 执行 前 ) 表 tb 
empid month jE 大 的 
> 50 的 
二 一 一 了 
A = 记录 为 对 象 ， 按 照 
empid 分 组 ， 并 降 
A104 a 序 显示 各 组 sales 
A101 4 的 平均 值 大 于 等 
A103 5 120 的 记录 
A101 9 
A102 6 
A104 局 
A103 6 
A107 6 
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总 结 
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区 





即 可 。 





HINT 
将 8.9.2 节 的 问题 2 转换 为 视 








ta 参考 答案 





问题 1 








执行 下 面 的 语句 。 


CREATE VIEW Vv sales 
AS SELECT 

empid,AVG(sales) 
FROM tb 


WHERE sales>=50 
GROUP BY empid 
HAVING AVG(sales) >=120 
ORDER BY AVG(sales) DESC; 








mysql> CREATE VIEW V_sales 


= AS SELECT 

-> empid,AVG (sales) 

-> FROM tb 

-> WHERE sales>=50 

-> GROUP BY empid 

-> HAVING AVG(sales)>=120 

-> ORDER BY AVG(sales) DESC; 


Query OK, 0 rows affected (0.08 sec) 


mysql> SELECT * FROM Vv _ sales; 


+------- a + 
| Alol | 242.0000 | 
| Al04 | 137.0000 | 
| A102 | 129.5000 | 
下 Fa + 


3 rows in set (0.00 sec) 
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使 用 存储 过 程 后 就 可 以 记录 一 系列 操作 并 批量 运行 它们 了 。 
“每 次 都 要 重复 相同 的 操作 ”“ 统 一 记录 一 系列 的 操作 ， 以 免 忘 
记 ” 等 情况 非常 适合 使 用 存储 过 程 处 理 。 

如 果 需 要 多 次 执行 相同 的 SQL 命令 ， 就 可 以 事先 把 这 个 处 
理 定义 为 存储 过 程 。 只 要 使 用 命令 “cALL x x”， 就 可 以 立即 
执行 定义 的 处 理 。 






























































12.1 什么 是 存储 过 程 


5 12.1.1 可 用 的 版 本 


在 学 习 存储 过 程 之 前 ， 请 先 确认 MySQL 的 版 本 (一 11.1.2 节 )。 事先 说 明 一 下 ， 存 储 过 程 只 能 
在 5.0 或 更 高 的 版 本 中 使 用 。 





山中 





CS 12.1.2 ”什么 是 存储 过 程 


将 多 个 SQL 语句 组 合成 一 个 只 需要 使 用 命令 “cALL x x” 就 能 执行 的 集合 ， 该 集合 就 称 为 
存储 过 程 ( stored procedure )。“ 存 储 ”( stored ) 表示 保存 ,“ 过 程 ”( procedure ) 表示 步 又。 也 就 是 
说 ， 存 储 过 程 是 将 一 系列 步骤 归纳 并 存储 起 来 的 集合 ( 见 图 12-1 )。 
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CREATE PROCEDURE 
存储 过 程 1() 





INSERT INTO tb1 ~ 
SELECT * FROM table2 …'; 


创建 存储 过 程 
SELECT * FROM table3 ….; 








CALL ”存储 过 程 1(); | J 
RN 存储 过 程 1 
INSERT INTO tb1 … 
SELECT * FROM table2 ~; 一 系列 操作 


SBEECI ROM :tale 


K 理 结果 1.… 
执行 一 系列 处 理 处 理 结果 2 


处 理 结果 3.… 














图 12-1 存储 过 程 
由 于 可 以 自动 执行 许多 事先 准备 好 的 命令 ， 所 以 处 理 效 率 很 高 。 但 是 ， 在 存储 了 重要 数据 的 数 
据 库 中 ， 执 行 没 有 经 过 充分 验证 的 存储 过 程 是 非常 危险 的 ， 这 一 点 需要 我 们 牢记 。 
存储 过 程 乍 一 看 让 人 觉得 有 些 难以 理解 ， 但 只 要 学 会 了 它 的 使 用 方法 ， 用 起 来 就 会 非常 方便 。 
笔者 就 经 常 使 用 CALL 来 代替 执行 多 次 SELECT。 
下 面 是 作为 示例 创建 的 存储 过 程 的 主体 。 


SELECT * FROM tb; 
SELECT * FROM tbl; 


上 面 的 示例 中 只 列 出 了 两 个 很 善 通 的 SQL 语句。 当然， 我 们 也 可 以 根据 需要 编写 任意 数量 的 
SQL 语句 ， 还 可 以 执行 编程 语言 中 常见 的 处 理 ， 比 如 使 用 变量 、 使 用 IF 和 CRAsE 作为 条 件 分 支 ， 
以 及 使 用 WHILE 和 REPEAT 反复 处 理 等 。 























12.2 ”使 用 存储 过 程 


® 12.2.1 创建 存储 过 程 


当 创 建 存储 过 程 时 ， 我 们 需要 像 下 面 这 样 执 行 CREATE PROCEDURE 命令 。 
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创建 存储 过 程 
; CREATE PROCEDURE 存储 过 程 名 () 
: BEGIN 
SQL 语句 1 
SQL 语句 2 
END 


从 BEGIN 到 END 为 止 的 内 容 是 存储 过 程 的 主体 。 
在 开头 加 上 BEGIN， 在 结尾 加 上 END， 这 么 做 可 以 明确 表示 “从 这 里 到 这 里 是 存储 过 程 的 命令 ”。 








说 ， 当 创建 前 一 节 介 绍 的 存储 过 程 时 ， 主 体 部 分 需要 像 下 面 这 样 描述 。 


BEGIN 
SELECT * FROM tb; 


SELECT * FROM tbl; 
END 





可 是 这 样 一 来 ， 在 创建 存储 过 程 的 时 候 就 会 输入 分 隔 符 “;”。 在 这 种 情况 下 ,，CREATE 
PROCEDURE 命令 就 会 在 存储 过 程 不 完整 的 状态 下 执行 。 因 为 在 MySQL 监视 器 中 一 旦 输入 了 分 隔 
符 ， 不 管 是 什么 内 容 ， 都 会 先 执行 分 隔 符 之 前 的 部 分 。 








P 修改 分 隔 符 的 设置 
在 存储 过 程 不 完整 的 状态 下 执行 命令 会 带 来 一 些 麻烦 ， 因 此 我 们 需要 改变 环境 设置 ， 在 输入 了 
最 后 的 END 之 后 再 执行 CREATE PROCEDURE 命令 。 
因此 ， 在 创建 存储 过 程 时 ， 需 要 事先 将 分 隔 符 从 “;” 修 改 为 其 他 符号 ， 一般 使 用 “//”。 
我 们 可 以 使 用 delimiter 合 令 将 分 隔 符 修改 为 “//”。 


将 分 隔 符 修改 为 “//” 














如 果 将 分 隔 符 设置 为 “/ /”"， 那 么 即使 在 创建 存储 过 程 的 途中 输入 了 “;” 也 不 会 发 生 任 何 问 
题 。 在 END 之 后 输入 “//”， 这 时 就 会 执行 CREATE PROCEDURE 命令 。 存 储 过 程 创 建 结束 后 ， 使 
用 “delimiter ;” 将 分 隔 符 恢复 为 原始 设置 。 

下 面 我 们 来 实际 操作 一 下 。 创 建 一 个 执行 “SELECT * FROM tb;” 和 “SELECT * FROM 
tb1; ”的 存储 过 程 prl。 




















P> 显示 表 tb 








执行 存储 过 程 pr1 














































































































empid sales month 
A103 101 4 
A102 54 5 
A104 181 4 
A101 184 4 
A103 17 5 
A101 300 5 
A102 205 6 
A104 93 5 
A103 12 6 
A107 87 6 
BA 
> 接着 显示 表 tb1 
empid name age 
A101 佐 爱 40 
A102 高 桥 28 
A103 中 川 20 
A104 渡 边 23 
A105 西 泽 35 
操作 方法 









































表 tb1 显示 了 出 来 


中 
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delimiter // 
CREATE PROCEDURE pr1() 


BEGIN 


SELECT * FROM tb; 
SELECT * FROM tbl; 


END 
// 


delimiter ; 





254 | 第 12 章 熟练 使 用 存储 过 程 





mysdql> delimiter // 
mysql> CREATE PROCEDURE Prl() 
-> BEGIN 
-> SELECT * FROM tb; 
-> SELECT * FROM tbl; 
-> END 
= 
Query OK, 0 rows affected (0.53 sec) 


mysql> delimiter ; 

















最 后 的 “delimiter;” wh “;” ee 要 忘记 恢复 分 隔 符 。 
另外 ， 一 定 要 在 存储 过 程 名 的 后 面 加 上 ()。 之 后 会 介 和 Dg 十 程 中 的 示例 
(一 下 一 节 )， 但 即使 不 输入 值 也 必须 加 上 () 


5 12.2.2 ”执行 存储 过 程 


这 样 就 创建 好 了 存储 过 程 prl 。 下 面 我 们 来 试 着 使 用 这 个 存储 过 程 。 执 行 存 储 过 程 需 要 使 用 


CALL 命令 。 














执行 存储 过 程 


调用 上 一 节 创 建 的 prl。 


CRALL PIL; 





执行 结果 

mysql> CALL prl 

站 二 下 = 十 
empid sales month 

+------- +------- +------- 十 
A1l03 101 4 
Al02 54 5 
Al04 二 8 于 4 
A1l01 184 4 
A1l03 17 S 
Al01 300 Ss 
A1l02 205 6 
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| A104 | 93 | 5 | 
| A103 | 12 | 6 | 
| A107 | 87 | 6 | 
i mt 和 二 


10 rows in set (0.16 sec) 



































起 二 i 中 
empid name age 

和 学 部 大 二 志和 * 
ARA101 佐 芯 40 
A102 高 村 28 
A103 中 川 20 
A1l04 渡 边 23 
RA105 西 泽 35 

二 夭 呈 和 笑 忆 半生 中 


5 rows in set (0.20 sec) 








Query OK, 0 rows affected (0.20 sec) 





成 功 地 连续 执行 了 “SELECT * FROM tb;” 和 “SELECT * FROM tbl;” 这 两 条 命令 。 


人 12.2.3 创建 只 显示 大 于 等 于 指定 值 的 记录 的 存储 过 程 























接 下 来 试 着 创建 带 参数 (一 8.2.2 节 ) 的 存储 过 程 。 将 需要 处 理 的 数据 指定 为 () 中 的 参数 ， 并 
执行 存储 过 程 。 

在 存储 过 程 中 ， 我 们 可 以 像 下 面 这样 编 写 参 数 。 

存储 过 程 中 参数 的 编写 


这 里 ， 我 们 来 创建 一 个 “显示 销售 额 大 于 等 于 指定 值 的 记录 ”的 基本 存储 过 程 。 

例如 ， 当 向 存储 过 程 pr 中 指定 整数 类 型 参数 d 时 ， 存 储 过 程 就 可 以 编写 为 PROCEDURE pr 
(a INT) 。 参 数 a 在 SQL 语句 中 的 编写 方式 与 普通 数值 相同 。 

例如 ， 当 处 理 下 面 这 种 情况 时 ， 























人 显示 表 tb 中 sales 大 于 等 于 参数 d 的 记录 
SQL 语句 可 以 编写 成 如 下 形式 。 


SELECT * FROM tb WHERE sales>=d; 
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下 面试 着 创建 存储 过 程 pr2。 如 果 指 定 整数 类 型 的 参数 a 的 值 执行 该 存储 过 程 ， 表 tb 中 销售 额 
大 于 等 于 da 的 记录 就 会 显示 出 来 。 


> 执行 内 容 

















旨 定 200 为 pr2 的 参数 来 执行 存储 过 程 











>、 4 作为 示例 ， 




















































































































> 表 tb 

empid sales month 

A103 101 4 

A102 54 5 

A104 181 4 

A101 184 4 

A103 17 5 

A101 300 5 

于 一 一 提取 sales 大 于 等 于 200 的 记录 

A102 206 6 

A104 93 5 

A103 12 6 

A107 87 6 
> 执行 后 

empid sales month 

A101 300 5 徊 一 一 一 一 一 显示 提取 的 记录 
A102 205 6 

操作 方法 

执行 下 面 的 命令 。 


delimiter // 

CREATE PROCEDURE pr2(d INT) 
BEGIN 

SELECT * FROM tb WHERE sales>=d; 
END 

// 

delimiter ; 
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mysql> delimiter // 
mysql> CREATE PROCEDURE pr2(d INT) 
-> BEGIN 
-> SELECT * FROM tb WHERE sales>=d; 
-> END 
-> // 
Query OK, 0 rows affected (0.02 sec) 





mysql> delimiter ; 











下 面 我 们 试 着 把 数值 200 指定 为 参数 来 执行 存储 过 程 pr2。 在 () 中 输入 200 并 执行 CALL。 


CALL pr2(200); 


mysql> CALL pr2(200); 


+------- +------- +------- + 








| empid | sales | month | 


2 rows in set (0.09 sec) 





Query OK, 0 rows affected (0.11 sec) 








于 是 ， 销 售 额 大 于 等 于 200 万 元 的 记录 显示 了 出 来 。 


在 参数 中 加 上 IN 的 例子 


如 果 按 照 上 述 操作 方法 使 用 参数 ， 在 参数 的 前 面 加 上 IN 也 会 得 到 相同 的 结果 ( IN d INT )。 相 反 ， 如 果 想 
将 处 理 结果 传 给 参数 ， 则 需要 加 上 oUT。 















































delimiter // 
CREATE PROCEDURE pr2 (IN d INT) 
BEGIN 


SELECT * FROM tb WHERE sales>=d; 
END 
// 


delimiter ; 
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12.3 显示、 删除 存储 过 程 


下 面 来 介绍 存储 过 程 的 显示 方法 和 删除 方法 。 




















5 12.3.1 显示 存储 过 程 的 内 容 


我 们 可 以 使 用 下 面 的 命令 来 显示 存储 过 程 的 内 容 。 
显示 存储 过 程 的 内 容 














例如 ， 下 面 显 示 了 存储 过 程 prl 的 内 容 。 




















2Ei 汪 ( 显示 存储 过 程 pr1 的 内 容 ) 





mysql> SHOW CREATE PROCEDURE prl; 





和 PE 二 一 
总 es 
= + 

| Procedure | sql mode | Create Procedure 

| character set client | collation connection Database Colla tion 
+----------- 人 +-- 
ee 机 
SEE 十 

| pri1 | STRICT TRANS TABLES,NO AUTO CREATE USER,NO ENGINE SUBSTITUTION | 
CREATE DEFINER= root @ localhost PROCEDURE “prl~() 

BEGIN 


SELECT * FROM tb; 
SELECT * FROM tbl1; 
END | gbk | gbk chinese ci | utf8 general ci 











1 row in set (0.00 sec) 








和 删除 数据 库 、 表 格 、 视 图 一 样 ， 我 们 可 以 使 用 DROP 命令 删除 存储 过 程 。 
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删除 存储 过 程 





全 ( 删除 存储 过 程 pr1 ) 





mysql> DROP PROCEDURE prl; 
Query OK, 0 rows affected (0.03 sec) 











12.4 什么 是 存储 函数 


12.4.1 可 用 版 本 
事先 说 明 一 下 ， 存 储 函 数 也 只 能 在 MySQL 5.0 或 更 高 的 版 本 中 使 用 。 

















12.4.2 ”什么 是 存储 函数 


如 图 12-2 所 示 ， 存 储 函 数 ( stored function ) 的 思考 方式 和 操作 方法 与 存储 过 程 基本 相同 。 与 
存储 过 程 唯一 不 同 的 一 点 是 ， 存 储 函 数 在 执行 后 会 返回 一 个 值 。 








CREATE FUNCTION 
存储 函数 1 .… 


INSERT INTO tb1 .. 
RETURN 计算 创建 存储 
加 函数 ! 








SELECT 存储 函数 1(); | 


SS 存储 函数 1.… 
RETURN 
RETURN 计算 » 








图 12-2 存储 函数 
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正如 它 的 名 字 所 表达 的 那样 ， 存 储 函 数 可 以 作为 函数 工作 。 虽 然 8.2.2 节 提 到 了 MySQL 中 有 
许多 函数 ,但 使 用 存储 函数 可 以 创建 自 定义 的 函数 。 所 以 存储 函数 也 称 为 用 户 自 定义 函数 。 

存储 函数 返回 的 值 可 以 在 SELECT 和 UPDATE 等 命令 中 和 普通 函数 一 样 使 用 。 

我 们 可 以 使 用 以 下 语句 创建 存储 函数 。 

































































创建 存储 函数 
CREATE FUNCTION 存储 函数 名 ( 参数 数据 类 型 ) RETURNS 返回 值 的 数据 类 型 
: BEGIN 
Semsane 
RETURN 返回 值 . 表达 式 
END 


和 存储 过 程 一 样 ， 我 们 可 以 在 () 内 指定 参数 。 即 使 没有 指定 参数 ， 也 必须 加 上 () 。 例 如 ， 当 
创建 了 名 为 fu 的 存储 函数 时 ，fu () 本 身 将 返回 存储 函数 中 RETURNx x 的 x x we。 














12.5 ”使 用 存储 函数 


如 果 在 本 书 介绍 的 环境 中 ?使 用 存储 函数 ， 在 使 用 之 前 需要 修改 一 处 设置 。 
存储 函数 有 可 能 对 复制 (一 11.5.3 节 ) 和 数据 的 恢复 产生 影响 。 因 此 ， 参 数 1og_bin_ 
trust function creators 的 初始 值 被 设置 为 0, 这 样 就 不 能 使 用 存储 函数 了 “。 要 想 使 用 存储 
就 需要 执行 下 面 的 操作 修改 此 设置 。 当 然 ， 在 本 书 介绍 的 范围 内 修改 这 个 设置 不 会 产生 任何 
可 题 ， 请 大 家 放心 。 





















































QD 这 里 指 开 启 了 二 进 制 日 志 功 能 的 环境 。 我 们 可 以 通过 “show variables like '1og bin'; ”命令 确 认 是 否 
开启 了 该 功能 。 

@) 在 开启 二 进 制 日 志 功 能 的 情况 下 ，1log_bin trust function creators 参数 用 于 控制 是 否 可 以 信任 存储 函 
数 创建 者 ， 防 止 创建 写 入 二 进 制 日 志 引 起 不 安全 事件 的 存储 函数 。 默 认 值 为 0 (OFF ) 表示 用 户 不 能 创建 或 修改 
存储 函数 ， 除 非 强 制 使 用 DETERMINISTIC、READS SQL DATA 或 NO SQL 特性 来 声明 函数 ， 明 确 告知 MySQL 
服务 器 这 个 函数 不 会 修改 数据 。 译 者 注 





译 者 注 
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操作 方法 





少 


Gd 启动 MySQL 监视 器 ， 执 行 下 面 的 命 


@ 确认 设置 是 否 已 正确 修改 ， 执 行 下 面 的 命令 。 








mysql> SET GLOBAL log bin trust function creators=1; 
Query OK, 0 rows affected (0.00 sec) 


mysql> SHOW VARIABLES LIKE 'log bin trust function creators'; 


| log bin trust function creators | ON | 
二- 一 +------- 十 


1 row in set (0.00 sec) 


























如 果 log bin trust function creators 被 设置 成 了 ON， 就 可 以 使 用 存储 函数 了 。 

另外 ， 如 果 重 新 启动 MySQL,，, log _ bin trust function creators 的 值 将 变 回 OFF。 
如 果 想 使 用 存储 函数 ， 就 需要 再 次 执行 上 述 操作 。 当 然 ， 也 可 以 通过 在 MySQL 的 配置 文件 中 或 者 
在 启动 MySQL 时 指定 相应 的 内 容 ， 让 log bin trust function creators 的 值 始终 为 ON。 
慎重 起 见 ， 本 书 没有 修改 初期 设置 ， 而 是 通过 修改 log_bin trust function creators 的 值 
进行 了 设置 。 

另外 ， 使 用 存储 函数 还 需要 相应 的 权限 。 大 家 使 用 的 root 用 户 中 已 经 包含 了 这 个 权限 ( Super 
权限 ), 但 是 普通 用 户 就 需要 额外 添加 权限 了 。 


















































光 看 文字 介绍 可 能 不 太 容 易 理 解 存储 函数 。 虽 然 和 数据 库 这 个 主题 不 太 相 符 ， 不 过 这 里 作为 练 
习 ， 我 们 来 试 着 计算 一 下 自己 的 标准 体重 。 
如 果 BMI=22 为 标准 体重 ， 则 有 如 下 等 式 。 














S 标准 体重 = 身高 (cm ) x 身高 ( cm ) x22 /10000 
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我 们 试 着 使 用 这 个 等 式 来 创建 存储 函数 ful。 
这 里 我 们 把 以 厘米 为 单位 的 身高 值 作为 参数 ， 并 指定 参数 名 为 neight， 类 型 为 整数 类 型 。 我 
们 需要 使 用 下 面 的 命令 在 存储 函数 ful 中 指定 INT 类 型 的 参数 。 








CREATE FUNCTION ful (height INT) 











存储 函数 自身 会 返回 值 。 因 此 ， 必 须 对 存储 函数 自身 返回 的 值 的 数据 类 型 进行 指定 。 存 储 函 数 
ful 返回 的 是 包含 小 数 的 标准 体重 ， 所 以 我 们 将 存储 函数 ful 返回 的 值 的 数据 类 型 指定 为 可 以 处 
理 小 数 部 分 的 DOUBLE 类 型 (一 5.2 节 )。 具 体 命令 如 下 所 示 。 


CREATE FUNCTION ful(height INT) RETURNS DOUBLE 


接着 在 存储 函数 中 ， 对 输入 的 height 值 取 平方 ， 再 乘 以 22， 然 后 除 以 10000， 最 后 通过 
RETURN 返回 计算 出 来 的 值 。 这 部 分 内 容 需 要 用 到 如 下 命令 。 


RETURN height * height *22/10000; 


下 面试 着 创建 一 个 自 定 义 的 标准 体重 处 理 函 数 。 创 建 把 以 厘米 为 单位 的 身高 值 指定 给 参数 
height 后 ， 就 能 返回 标准 体重 的 存储 函数 ful。 











PP 执行 内 容 


SELECT ful(174); 


将 身高 174 cm 指定 为 参数 、 4 


计算 174 x174 x22/10000 


1 


以 BMI 值 为 22 来 计算 、 4 


























返回 计算 后 的 值 ， 即 66 .6072 




















返回 标准 体重 
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操作 方法 




















delimiter // 

CREATE FUNCTION ful (height INT) RETURNS DOUBLE 
BEGIN 

RETURN height * height *22/10000; 

END 

1 


delimiter ; 





mysql> delimiter // 
mysql> CREATE FUNCTION ful (height INT) RETURNS DOUBLE 
-> BEGIN 
-> RETURN height * height *22/10000; 
-> END 
= 
Query OK, 0 rows affected (0.00 sec) 


mysql> delimiter ; 























下 面 我 们 试 着 按 身 高 174 cm ( 笔者 的 身高 ) 来 计算 一 下 标准 体重 。 这 次 让 ful 作为 函数 来 返 





回 值 。 因 此 我 们 不 使 用 cALL， 而 是 使 用 SELECT 命令 来 显示 ful () 的 值 。() 内 输入 的 参数 值 为 


174。 


SELECT ful(174) ; 


mysql> SELECT ful(174); 





EE + 
| Eul(174) | 
a + 
| 66.6072 | 
二 二 


1 row in set (0.00 sec) 











以 笔者 的 身高 为 例 ， 标 准 体重 应 为 66.6 kg。 
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试 着 创建 一 个 存储 函数 来 返回 某 列 中 数据 的 平均 值 。 事 先 创建 好 这 样 的 函数 ， 使 用 的 时 候 就 会 非常 方便 。 
以 下 内 容 用 于 创建 存储 函数 Eu2 来 返回 表 tb 中 列 sales 的 平均 值 。 





CREATE FUNCTION fu2 () RETURNS DOUBLE 
BEGIN 


END 





编程 经 验 少 的 人 可 能 不 太 明 白 变 量 的 相关 人 处理， 这 里 我 们 先 创建 好 这 个 存储 函数 。 变 量 的 含义 
将 在 后 面 进行 介绍 。 
国 的 部 分 与 前 一 节 一 样 ， 它 用 于 定义 存储 函数 fu2 的 返回 值 为 DOUBLE 类 型 。 


CRERATE FUNCTION fu2 () RETURNS DOUBLE 


平均 值 需要 通过 SELECT RAVG () 来 计算 。 这 个 值 必须 赋 给 变量 。 我 们 可 以 把 变量 理解 为 “ 保 
管 值 的 箱子 ”"。 要 想 使 用 变量 ， 就 需要 事先 通过 DECLARE 定义 变量 。 

















通过 DECLARE 定义 变量 


这 里 我 们 把 变量 名 设置 成 了 上 。 平 均值 会 输入 到 变量 * 中 。 为 了 能 够 处 理 小 数 部 分 ， 我 们 将 数 
据 类 型 指定 为 DOUBLE 类 型 。 将 变量 上 定义 为 DOUBLE 类 型 的 命令 为 加 所 指 的 部 分 ， 即 


DECLARE r DOUBLE; 


下 面 的 SQL 语句 用 于 从 表 tb 中 提取 列 sales 的 平均 值 (一 8.2.2 节 )。 


SELECT AVG(sales) FROM tb; 


把 AVG (sales) 赋 给 DECLARE 中 定义 的 变量 r 时 需要 使 用 INTO。 于 是 ，SQL 语句 就 变 成 了 
图 所 指 的 部 分 ， 即 


SELECT AVG(sales) INTO r FROM tb; 


这 样 ， 平 均值 就 输入 到 了 变量 上 中。 我 们 可 以 使 用 圆 的 RETURN 让 这 个 平均 值 作为 存储 也 数 的 值 
返回 。 
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RETURN rr; 


我 们 来 实际 操作 一 下 。 试 着 创建 存储 函数 fu2 ， 将 表 tb 中 列 sales 的 平均 值 作为 DOUBLE 类 型 
的 值 返回 。 































































































>、 4 SELECT 存储 函数 fu2 
PF 表 tb 
empid sales month 
A103 101 因 几 一 一 一 一 一 一 一 计算 sales 的 平均 值 
A102 54 |s 
A104 181 4 
A101 184 4 
A103 1 5 
A101 300 5 
A102 取 05 6 
A104 93 5 
A103 [| 谤 |6 
A107 87 le 




















显示 平均 值 123.4@ 一 一 一 一 一 计算 AvVG (sales) 
































delimiter // 

CREATE FUNCTION fu2() RETURNS DOUBLE 
BEGIN 

DECLARE r DOUBLE; 

SELECT AVG(sales) INTO r FROM tb; 
RETURN r; 

END 

// 

delimiter ; 
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mysdql> delimiter // 
mysql> CREATE FUNCTION fu2() RETURNS DOUBLE 
-> BEGIN 
-> DECLARE r DOUBLE; 
-> SELECT AVG(sales) INTO r FROM tb; 
-> RETURN r; 
-> END 
-> // 
Query OK, 0 rows affected (0.05 sec) 


mysql> delimiter ; 











创建 好 存储 函数 fu2 后 ， 试 着 使 用 SELECT 命令 显示 平均 值 。 


SELECT fu2(); 





1 row in set (0.06 sec) 











平均 值 123.4 显示 了 出 来 。 





全 12.5.4 显示 和 删除 有 储 B 数 


存储 函数 的 显示 方法 和 删除 方法 与 存储 过 程 相同 (一 12.3 节 )。 
i 
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执行 结果 

mysql> SHOW CREATE FUNCTION fu2; 

+---------- +---------- +----- 
--+---------------------- +---------------------- +-------------------- + 


| Function | sql mode | Create Function 


| character set client | collation connection | Database Collation | 
+---------- +---------- +----- 
--+---------------------- +---------------------- +-------------------- 十 

| Eu2 | | CREATE DEFINER= “root“@ “localhost” FUNCTION “~fu2 ”~ () 
RETUR 

NS double 

BEGIN 


DECLARE r DOUBLE; 

SELECT AVG (sales) INTO r FROM tb; ®@ 一 一 一 一 一 一 主体 部 分 的 描述 
RETURN rr; 
END | gbk | gbk chinese ci utf8 general ci 




















1 row in set (0.00 sec) 
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12.6.1 什么 是 触发 器 


(CO 


触发 需 〈trigger ) 是 一 种 对 表 执 行 某 操作 后 会 触发 执行 其 他 命令 的 机 制 。 
当 执行 INSERT、UPDRATE 和 DELETE 等 命令 时 ， 作 为 触发 右 提 前 设置 好 的 操作 也 会 被 执行 。 





例如 ， 创 建 一 个 触发 器 ， 当 某 表 的 记录 发 生 更 新 时 ， 就 以 此 为 契机 将 更 新 的 内 容 记 录 到 另 一 个 





表 和 


PhP ( 见 图 12-3 )。 
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以 此 为 契机 表 1 


列 1 ， 列 ?2 ， 列 3 | 























DELETE 
INSERT CE 更 新 前 OLD. 列 
UPDATE 














表 2 


1 ， 列 2 ， 列 3 | 
开始 处 理 | | | 






































图 12-3 触发 器 
触发 器 也 常 作为 处 理 的 记录 或 者 处 理 失败 时 的 备份 使 用 。 
触发 需 是 一 个 非常 强大 的 功能 ， 只 通过 文字 介绍 很 难 了 解 到 这 个 功能 有 哪些 优点 。 我 们 先 创建 
一 个 触发 吉 来 体验 一 下 。 
这 里 我 们 来 创建 一 个 “如 果 删 除了 表 中 的 记录 ， 被 删除 的 记录 就 会 复制 到 其 他 表 中 ”的 触发 









































器 。 也 就 是 说 ， 如 果 对 表 tbl 执行 “DELETE FROM tbl ...;” 命 令 ， 被 删除 的 记录 就 会 全 部 插入 
到 表 tbl_ from 中 。 这 样 一 来 ， 我 们 就 可 以 随时 恢复 已 删除 的 记录 了 。 

请 事先 创建 一 个 空 表 tp1_from 用 于 插入 表 tbl 中 删除 的 记录 。 我 们 可 以 通过 复制 表 tbl 的 列 结 
构 来 创建 这 个 表 (一 7.3.1 节 )。 


CREATE TABLE tbl] from LIKE tb1， 


12.7 创建 触发 器 


触发 需 会 在 对 表 执 行 INSERT、UPDATE 和 DELETE 等 命令 之 前 ( BEFORE ) 或 之 后 (AFTER ) 
被 调用 和 执行 。 





























触发 天 被 触发 的 时 机 包括 以 下 两 种 。 
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> 触发 器 被 触发 的 时 机 
BEFORE 在 对 表 进 行 处 理 之 前 触发 
AFTER 在 对 表 进 行 处 理 之 后 触发 















































另外 ， 对 表 进 行 处 理 之 前 的 列 值 和 对 表 进 行 处 理 之 后 的 列 值 ， 可 以 像 下 面 这 样 通过 “oLD . 列 
名 ”“NEW. 列 名 ”获得 。 


























> 列 值 
OLD. 列 名 对 表 进 行 处 理 之 前 的 列 值 
NEW. 列 名 对 表 进 行 处 理 之 后 的 列 值 





























也 就 是 说 ， 执 行 INSERT、UPDATE 和 DELETE 命令 之 前 的 列 值 可 以 通过 “oLD . 列 名 ”获得 ， 
执行 这 些 命令 之 后 的 列 值 可 以 通过 “NEW. 列 名 ”获得 。 

但 是 ， 根 据 命 令 的 不 同 ， 有 的 列 值 可 以 取出 来 ， 有 的 列 值 不 能 取出 来 。 在 表 12-1 中 ，O 〇 表示 
可 以 取出 来 的 列 值 。 








表 12-1 触发 器 执行 前 后 ， 列 值 是 否 可 以 取出 来 








执行 前 ( OLD. 列 名 ) ※ 使 用 BEFORE ENA 
INSERT x O 
DELETE O x 











O 


UPDATE O 














S 12.7.2 ”创建 触发 器 


实际 操作 一 下 会 更 容易 理解 。 创 建 触发 需 的 具体 命令 如 下 所 示 。 
创建 触发 器 


; CREATE TRIGGER 触发 器 名 BEFORE |( 或 者 AFTER ) DELETE 等 命令 
; ON 表 名 FOR EACH ROW 
: BEGIN 

使 用 更 新 前 ( OLD. 列 名 ) 或 者 更 新 后 ( NEW. 列 名 ) 的 处 理 




















在 触发 右 主 体 的 描述 中 ， 各 个 命令 的 末尾 需要 加 上 “;”。 与 创建 存储 过 程 时 相同 ， 我 们 需要 事 
先 将 分 隔 符 改 为 “//” 等 (一 12.2.1 市 )。 
下 面 来 创建 将 表 tbl 中 删除 的 记录 插入 到 表 也 1_from 中 的 触发 器 。 
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PP 执行 内 容 
创建 将 表 tb1 中 删除 的 记录 插入 到 表 tb1_from 中 的 触发 器 


D4 























> 表 tb1 

empid name age 

A101 佐 且 40 

A102 高 桥 28 

A103 中 川 20 

A104 渡 边 23 

居 一 一 一 一 一 删除 表 tb1 的 记录 

A105 泽 35 


















































> 执行 删除 操作 后 的 表 tb1 














empid name age 
A101 佐 爱 40 
A102 高 桥 28 
A103 中 川 20 



































上 一 删除 的 记录 


区 ( 执行 后 ) 将 删除 的 记录 插入 到 表 tb_from 中 


empid name age 


A104 渡 边 23 
下 二 一 任何 时 候 都 可 以 恢复 ! 
A105 泽 35 




















他 
























































执行 下 




















操作 方法 
的 命令 。 


delimiter // 


CREATE TRIGGER trl1 BEFORE DELETE ON tb1 FOR EACH ROW 
BEGIN 


INSERT INTO tbl1 from VALUES (OLD.empid,OLD.name,OLD.age); 
END 


Lh 


delimiter ;，; 
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mysql> delimiter // 
mysql> CREATE TRIGGER trl1 BEFORE DELETE ON tbl FOR EACH ROW 
-> BEGIN 
-> INSERT INTO tbl from VALUES (OLD.empid,OLD.name,OLD.age); 
-> END 
-> // 
Query OK, 0 rows affected (0.08 sec) 





mysql> delimiter ; 











触发 需 创 建成 功 了 。 具 体内 容 将 在 12.7.3 节 进 行 介 绍 。 我 们 先 来 体验 一 下 触发 器 的 效果 。 
现在 ， 表 tbl 中 删除 的 记录 应 该 能 够 插入 到 表 tb1_from 中 了 。 不 管 是 1 条 记录 还 是 2 条 记录 ， 
触发 器 都 会 进行 处 理 ， 所 以 我 们 干脆 把 所 有 记录 都 删 掉 。 


DELETE FROM tb1; 


首先 试 着 使 用 “SELECT * FROM tb1;” 命 令 确认 记录 是 和 否 真 的 被 删除 了 。 


mysql> DELETE FROM tbl; 
Query OK, 5 rows affected (0.14 sec) 




















mysql> SELECT * FROM tbl; 
Empty set (0.00 sec) 


























执行 结果 中 显示 了 Empty set， 没 有 显示 任何 记录 。 表 tbl 中 已 经 没有 记录 了 。 那 么 ,设置 
的 触发 需 是 否 正 常 工作 了 呢 ? 下 面 ， 我 们 来 看 一 下 表 tbl_from 中 的 内 容 。 


SELECT * EROM tb1l fromy” 


mysql> SELECT * FROM tbl1 from; 









































Ss es Ss + 
empid name age 
ce 趟 二 和 二 + 
A1l01 佐 芯 40 
A102 高 村 28 
A1l03 中 川 20 
ARA104 渡 边 23 
RA105 西 泽 35 
于 和 中富 训 训 总 总 各 + 





5 rows in set (0.05 sec) 
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和 我 们 预想 的 一 样 ， 删 除 的 记录 插入 到 了 表 tbl from 中 。( sELECT 显示 的 顺序 可 能 会 发 生 
改变 。) 由 此 可 见 ， 触 发 器 正常 工作 了 。 在 没有 正常 工作 的 情况 下 ， 请 确认 一 下 输入 历史 。 大 多 
是 编写 错误 所 致 。 如 12.8.1 节 所 示 ， 我 们 可 以 使 用 SHOW 命令 来 确认 触发 器 的 内 容 。 如 果 发 现 触 
发 器 trl 的 内 容 有 问题 ， 请 使 用 “DROP TRIGGER trl;” 了 予以 删除 (一 12.8.2 节 )， 然 后 重新 
创建 触发 器 。 

下 面 将 插入 到 表 tbl_from 中 的 记录 恢复 到 原来 的 表 tbl 中 。 我 们 可 以 使 用 下 面 的 命令 进行 复制 
(一 7.5 节 )。 


INSERT INTO tb1 SELECT * FROM tbl1 from; 


12.7.3 触发 器 的 内 容 


下 面 看 一 下 触发 吉 的 内 容 。 



































CREATE TRIGGER trl1 BEFORE DELETE ON tb1l FOR EACH ROW 
BEGIN 


INSERT INTO tb1 from VALUES (OLD.empid,OLD.name,OLD.age); 
END 
// 





首先 为 表 tbl 的 DELETE 命令 设置 触发 器 trl。 因 为 要 对 删除 之 前 (BEFORE) 的 值 进行 
INSERT， 所 以 CREATE TRIGGER 的 部 分 需要 编写 成 下 面 这 样 。 

















CREATE TRIGGER trl1 BEFORE DELETE ON tbl1 FOR EACH ROW 








提取 删除 记录 前 (BEFORE ) 的 列 值 (oLD. 列 名 )， 并 将 其 插入 表 tbl_from 中 。 表 tbl_from 
由 列 empid、 列 name 和 列 age 组 成 ， 所 以 删除 记录 前 的 列 值 分 别 为 OLD.empid、OLD.name 和 
OLD.age。 

因为 要 使 用 INSERT 命令 将 这 些 列 值 插入 到 表 tbl_from 中 ， 所 以 触发 器 需要 描述 成 下 耳 














oO 
INSERT INTO tbl1 from VALUES (OLD.empid,OLD.name,OLD.age); 


该 触发 器 的 主体 部 分 用 BEGIN 和 END 括 了 起 来 。 
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12.8 ”确认 和 删除 触发 器 


5 12.8.1 确认 设置 的 触发 器 


触发 器 是 自动 启动 的 。 为 了 避免 无 意 间 执行 相关 处 理 ， 我 们 需要 在 管理 上 多 加 注意 。 必 须 对 当 
前 设置 的 触发 器 及 其 内 容 有 充分 的 了 解 。 
我 们 可 以 使 用 SHOW TRIGGERS 命令 确认 当前 设置 的 触发 髓 。 


确认 触发 器 





























mysql> SHOW TRIGGERS; 





A 
+--------- +-------- +------- 二 一 
---------------- 十 = 一 一 一 
-------------------- 二 一 
-+ + 

| Trigger | Event | Table | Statement 





| Timing | Created | sql mode 
| Definer | character set client | collation connecti 
on | Database Collation | 








本 ee 和 Se 
3 sd te Di et 
3 td 
i + 

| tri1 | DELETE | tpb1l | BEGIN 

INSERT INTO tbl1 from VALUES (OLD.empid,oOLD.name,OLD.age); 

END | BEFORE | NULL | STRICTTIRANS TABLES,NO AUTO CREATE USER,NO ENGINE 
SUBS$T 

ITUTION | root@localhost | gbk | gbk chinese ci | utf8 ge 
neradl ci 

机 Se es 人 i 
a st ta tt 
es tt ee ti 
OE OTE TOPO ERR 有 


1 row in set (0.14 sec) 























触发 器 tr1 触发 器 tr1 的 主体 




















274 | 第 12 章 熟练 使 用 存储 过 程 


为 了 防止 意外 执行 处 理 ， 我 们 需要 删除 不 需要 的 触发 顺 。 
删除 触发 器 


试 着 删除 触发 器 trl。 请 执行 以 下 操作 。 


DROP TRIGGER trl; 


执行 结果 





mysql> DROP TRIGGER trl; 
Query OK, 0 rows affected (0.05 sec) 














这 样 ， 触 发 需 trl 就 被 删除 了 。 试 着 通过 “sHOW TRIGGERS; ”命令 进行 确认 。 


执行 结果 





mysql> SHOW TRIGGERS 
Empty set (0.05 sec) 


本 章 介 绍 了 以 下 内 容 。 











@ 存 储 过 程 的 含义 和 创建 方法 

@ 存 储 函 数 的 含义 和 创建 方法 

@ 存 储 过 程 和 存储 函数 中 参数 的 处 理 方法 

@ 触 发 器 的 含义 和 创建 方法 

@ 触 发 器 能 够 提取 的 列 的 数据 种 类 和 触发 时 机 

我 们 知道 了 存储 过 程 是 将 多 个 处 理 统一 执行 的 集合 。 如 果 利 用 存储 过 程 统 一 执行 服务 器 上 的 处 

， 就 可 以 减少 客户 端 和 服务 右 之 间 的 通信 。 也 就 是 说 ， 存 储 过 程 可 以 提高 整个 处 理 过 程 的 效率 。 
事先 将 处 理 汇总 到 一 起 能 够 有 效 防 止 执行 顺序 发 生 错误 。 








> 自我 检查 





下 面 检查 一 下 本 章 学 习 的 内 容 是 否 全 部 型 








解 并 掌握 了 。 


12.9 总 结 | 275 


口 能 够 使 用 CREATE PROCEDURE 创建 存储 过 程 并 使 用 cALL 来 执行 
口 能够 自由 地 更 改 分 隔 符 
口 能够 创建 带 参数 存储 过 程 
口 能 够 创建 存储 函数 
口 能 够 创建 返回 sELECT 的 值 的 存储 函数 

口 能 够 使 用 CREATE TRIGGER 创建 触发 器 

口 能 够 通过 BEFORE、AFTER 提取 执行 前 和 执行 后 的 列 值 
























































































































































































































































i 练习 题 
问题 1 
请 使 用 下 面 的 表 tb， 创 建 如 果 指 定 4 ~ 6 月 任意 一 个 月 ( month ) 的 数值 给 参数 上 t ( INT 类 型 )， 就 返 
可 该 月 销售 额 ( sales ) 总 和 ( INT 类 型 ) 的 存储 函数 £_sales。 
忆 表 tb 
empid sales month 
A103 TOi 4 
A102 54 5 
A104 181 4 
A101 184 4 
A103 Ml 5 
A101 300 三 
A102 205 6 
A104 98 B 
A103 和 2 6 
A107 87 6 
“ 这 是 一 个 实用 的 存储 函数 。 试 着 回想 一 下 参数 和 内 部 处 理 中 使 用 的 变量 的 设置 方法 ， 以 及 
Le 更 用 RETURN 返回 值 的 方法 。 
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局 参考 答案 








问题 1 








执行 下 面 的 命令 。 


delimiter // 

CREATE FUNCTION f sales(t INT) RETURNS INT 
BEGIN 

DECLARE U INT; 


SELECT SUM(sales) INTO u FROM tb WHERE month=t; 
RETURN u; 

END 

// 


delimiter ; 








mysql> delimiter // 
mysql> CREATE FUNCTION f£ sales(t INT) RETURNS INT 
-> BEGIN 
-> DECLARE u INT; 
-> SELECT SUM(sales) INTO u FROM tb WHERE month=t; 
-> RETURN u; 
-> END 
-> // 
Query OK, 0 rows affected (0.05 sec) 


mysql> delimiter ; 
mysql> SELECT f£ sales(4); 


1 row in set (0.02 sec) 











如 果 无 法 执行 命令 ， 请 确认 变量 1og bin trust function creators 的 值 是 否 设置 成 了 
1 (12.5:1 人 六 
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将 一 系列 操作 作为 单个 逻辑 工作 单元 处 理 的 事务 ， 是 数据 库 
过 程 中 不 可 或 缺 的 功能 。 本 章 将 结合 MySQOL 独 有 的 存储 引 


擎 的 设置 对 事务 进行 介绍 。 


















































六 














13.1 什么 是 存储 引擎 


13.1.1 什么 是 存储 引擎 


在 学 习 事务 之 前 ， 我 们 需要 了 解 一 下 MySQL 中 的 存储 引擎 (storage engine )。 

MySQL 的 功能 大 致 分 为 两 种 。 

一 个 是 连接 客户 端 和 提前 检查 SQL 语句 内 容 的 功能 ， 即 数据 库 处 理 的 “前 台 ” 部 分 。 另 一 个 
是 根据 前 台 部 分 的 指示 ， 完 成 查询 和 文件 操作 等 工作 的 功能 ， 即 “后 台 ” 部 分 。 这 个 后 台 部 分 称 为 
存储 引擎 ( 见 图 13-1 )。 























MySOL 








于 提前 检查 SQL 语句 的 内 容 









存储 引擎 | 文件 1 
“ 查询 一 > 文件 2 
文件 操作 < 文件 3 








| MylSAM 和 InnoDB || csv |[MEMORY |… 





图 13-1 存储 引擎 
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S 13.1.2 存储 引擎 的 种 类 


MySQL 中 预 置 了 多 个 存储 引擎 ， 用 户 可 以 根据 使 用 目的 和 个 人 喜好 进行 选择 。 另 外 ， 每 个 表 
都 可 以 单独 指定 存储 引擎 ， 比 如 “A 表 是 OO 存储 引擎 ，B 表 x x 存储 引擎”。 

存储 引擎 相互 独立 且 人 允许 用 户 自 主 进行 选择 ， 正 是 MySQL 的 特征 。 

目前 MySQL 中 可 以 使 用 的 存储 引擎 主要 有 以 下 几 种 〈 见 表 13-1 )。 


表 13-1 MySQL 中 可 以 使 用 的 主要 存储 引擎 


存储 引擎 特征 




























































































































































































































































































MylSAM 以 前 版 本 中 的 默认 存储 引擎 。 虽 然 能 够 高 速 运行 ， 但 不 支持 事务 和 外 键 

InnoDB MySQL 5.5 或 更 高 版 本 的 默认 存储 引擎。 是 唯一 一 个 支持 事务 的 存储 引擎 。 本 书 使 用 的 就 是 这 个 存储 引擎 

BLACKHOLE | 写 入 的 任何 数据 都 会 消失 ， 查 询 始终 返回 空 结 果 。 于 复制 

MERGE 将 多 个 MylSAM 表 作 为 一 个 表 进 行 处 理 。 也 称 为 MRG_MylSAM 

CSV 将 数据 的 实体 保存 为 CSV ( 逗号 分 隔 ) 格式 的 文本 文件 。 该 文件 可 以 直接 通过 Excel 等 打 

EGRY 天 为 数据 全 部 存储 在 了 内 存 中 ， 所 以 处 理 速 度 非常 快 。 主 要 作为 从 临时 工作 区 和 其 他 表 中 提取 的 数 
所 的 读 取 专 用 缓存 使 用 。 不 支持 事务 

ARCHIVE 虽然 可 以 通过 压缩 来 存储 大 量 数 据 ， 但 是 仅 支 持 INSERT 和 SELECT 
































本 书 使 用 了 默认 存储 引擎 InnoDB。 事 实 上 ， 到 MySQL 5.4 为 止 的 版 本 中 ， 默 认 存 储 引擎 都 是 
MyISAM。 虽然 MyISAM 比 当时 的 InnoDB 的 人 处理 速度 快 ， 但 遗憾 的 是 它 不 支持 事务 。 因 此 ， 当 时 
也 出 现 了 “不 使 用 事务 的 话 就 选 MyISAM， 使 用 事务 的 话 就 选 mnoDB” 这 样 的 使 用 方式 。 但 现在 
InnoDB 的 处 理 速度 也 变 得 很 快 了 ， 所 以 我 们 不 需要 考虑 使 用 MyISAM。 

如 果 使 用 方法 比较 复杂 ， 就 需要 对 每 一 个 存储 引擎 进行 更 为 细致 的 调 优 ， 同 时 还 要 研究 新 的 存 
储 引 擎 。 但 本 书 以 夯实 基本 功 为 目标 ， 所 以 现 阶段 使 用 InnoDB 即 可 。 

不 过 ,存储 引擎 的 功能 正在 迅速 发 展 。 从 使 用 MySQL 的 角度 来 看 ， 我 们 需要 选择 最 适合 的 存 
储 引 擎 。 所 以 我 们 要 掌握 更 改 存储 引擎 的 方法 以 备 将 来 使 用 。 

































































13.2 ”设置 存储 引擎 


本 书 使 用 的 是 默认 设置 的 存储 引擎 hnoDB。 如 果 使 用 了 MySQL 5.4 或 54 之 前 的 版 本 ， 或 者 
使 用 了 其 他 存储 引擎 ， 就 需要 修改 设置 。 在 这 种 情况 下 ， 可 以 参考 操作 手册 等 修改 myini 的 设置 ， 
使 InnoDB 变 得 可 用 。 
ES 13.2.1 ”确认 存储 引擎 

下 面试 着 确认 一 下 我 们 用 过 的 表 的 存储 引擎 。 显 示 表 的 详细 信息 需要 使 用 SHOW CREATE 


TABLE 命令 。 
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SHOW CREATE TABLE tb; 


mysql> SHOW CREATE TABLE tb; 





| tb | CREATE TABLE “tb ( 

“empid ~ varchar(20) DEFAULT NULL, 
“sales” int(11) DEFAULT NULL 
“month” int(11) DEFAULT NULL 
) ENGINE=InnoDB DEFAULT CHARSET=utf8 | 





’ 

















1 row in set (0.25 sec) 


























我 们 可 以 在 ENGINE= x x 的 部 分 确认 存储 引擎 。 通 过 执行 结果 可 知 ， 在 上 面 的 示例 中 ， 表 也 
使 用 了 InnoDB。 当 创建 表 时 ， 尤 其 在 未 指定 存储 引擎 的 情况 下 ， 将 会 选择 默认 的 InnoDB。 





使 用 AG” 代 才 人 


像 “SHOW CREATE TABLE tb;” 这 样 的 命令 ，1 行 中 会 显示 很 多 数据 ， 所 以 易 读 性 较 差 。 这 时 可 以 在 命 
令 的 末尾 使 用 “\G” 代 蔡 “;”。"G” 一 定 要 大 写 。 
使 用 “\G” 之 后 ， 显 示 出 来 的 结果 就 能 像 下 面 这 样 按 列 纵向 显示 ， 更 易于 阅读 。 
















































































mysql> SHOW CREATE TABLE tb \G 
火炎 火炎 火炎 火炎 炎炎 火炎 大 火炎 大 大 大 大 炎炎 火炎 大大 大 大 le 
Table: tb 
Create Table: CREATE TABLE ‘tb ( 
“empid varchar(20) DEFAULT NULL, 


IOW 类 火 火 淡淡 火炎 火 火 火炎 炎炎 类 火炎 火炎 炎炎 火 火 火 火 火炎 类 


Sales Tn DRAUETI NOL 
“month int (11) DEFAULT NULL 
KEY ‘my ind (“empid ) 
) ENGINE=InnoDB DEEFRAULT CHRARSET=utf8 
1 row in set (0.00 sec) 





























在 SELECT 等 命令 中 ， 当 列 的 数量 较 多 或 者 显示 的 项 目 较 长 时 ， 这 个 技巧 就 会 发 挥 出 很 大 的 作用 。 请 大 家 
务必 记 住 这 个 技巧 。 
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13.2.2 ”修改 存储 引擎 


一人 一 一 一 一 一 一 


表 的 存储 引擎 可 以 修改 。 作 为 示例 ， 我 们 试 着 将 表 tb1B 的 存储 引擎 由 InnoDB 修改 为 


MyISAM, 








我 们 可 以 使 用 ALTER TABLE 命令 修改 存储 引擎 。 








将 存储 引擎 修改 为 MylSAM 





mysql> ALTER TABLE tblB ENGINE=MyISAM; 
Query OK, 9 rows affected (0.91 sec) 


Records: 9 Duplicates: 0 Warnings: 0 














即使 显示 了 Query OK， 存储 引擎 也 可 能 没有 被 修改 。 所 以 一 定 要 执 
TABLE tblB;” 来 确认 存储 引擎 已 被 成 功 修改 为 MyISAM。 


六 二 认 


人 SHOW 


CREATE 





执行 结果 

mysql> SHOW CREATE TABLE tbl1B; 

灶 呈 和 二 生 二 学 二 A 
--------- 十 


| tblB | CREATE TABLE “~tblb ”~ ( 
“empid™ varchar(10) DEFAULT NULL, 
“name™ varchar (10) DEFAULT NULL, 
~age ”int(11) DEFAULT NULL 

) ENGINE=MyISAM DEFAULT CHARSET=utf8 | 























1 row in set (0.03 sec) 
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13.3 ”什么 是 事务 


在 介绍 事务 的 时 候 经 常会 提 到 的 一 个 例子 就 是 存 取款 处 理 。 
假设 hoge 先生 想 把 账户 上 的 10 万 元 转 到 piyo 先生 的 账户 上 。 这 时 就 需要 从 
余额 中 扣除 10 万 元 ， 给 piyo 先生 的 账户 余额 加 上 10 万 元 。 



































过 
7 





如 果 从 hoge 先生 的 账户 余额 中 扣除 10 万 元 时 发 生 了 错误 ,会 出 现 什 么 样 的 情 


hoge 先生 的 账户 


况 ? 结果 就 是 在 











没有 将 10 万 元 转 给 piyo 先生 的 状态 下 ，hoge 先生 账户 中 的 10 万 元 不 经 而 飞 ( 见 
Te 6 在 存 取 款 处 理 中 出 现 。 


图 13-2 )。 这 种 情 


“扣除 10 万 元 ”和 “增加 10 万 元 ”的 操作 应 该 作为 一 个 不 可 分 割 的 整体 来 处 理 。 如 果 
es ss 的 操作 失败 ， 那 么 “扣除 10 万 元 ”的 操作 也 应 该 取消 掉 ( 见 图 13-3 )。 这 样 至 少 可 





以 避免 10 万 元 赁 空 消失 的 情况 发 生 。 








如 上 所 述 ， 将 多 个 操作 作为 单个 逻辑 工作 单元 处 理 的 功能 称 为 事务 ( transaction )。 将 事务 开始 
之 后 的 处 理 结果 反映 到 数据 库 中 的 操作 称 为 提交 (commit )， 不 反映 到 数据 库 中 而 是 恢复 为 原来 状 





态 的 操作 称 为 回 滚 (rollback )。 





习惯 了 个 人 计算 机 的 处 理 之 后 ， 经 常会 觉得 只 要 按 下 Ctrl+tZ， 就 能 恢复 原状 (UNDO )。 但 是 在 








数据 库 的 世界 中 ， 除 非 开 启 了 事务 处 理 ， 和 否则 数据 修改 之 后 是 无 法 恢复 原状 的 。 


姓名 ; ”金额 


hoge ! 

































































































! —100,000 









































图 13-2 存 取 款 处 理 失 败 导 致 资金 消失 会 很 麻烦 ! 


如 果 发 生 错误 


遇 到 麻烦 
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图 13-3 事务 


13.4 ”使 用 事务 


下 面试 着 体验 一 下 事务 的 功能 。 这 里 ， 我 们 删除 销售 信息 表 tb 中 的 所 有 记录 ， 然 后 试 着 执行 
回 滚 处 理 。 

















13.4.1 执行 前 的 注意 事项 


删除 重要 的 数据 后 如 果 无 法 复原 就 麻烦 了 ， 所 以 在 做 事务 的 相关 练习 时 ， 一 定 要 使 用 无 关 紧 要 的 
数据 。 此 外 ，DROP 等 命令 是 不 能 执行 回 滚 操作 的 (无 法 复原 )， 这 一 点 需要 大 家 注意 (一 13.5.3 节 )。 

另外 ,存储 引擎 为 MyISAM 的 表 无 法 使 用 事务 功能 。 确 认 表 tb 的 存储 引擎 是 默认 的 InnoDB 
之 后 再 使 用 事务 (一 13.2.2 节 )。 



































S 13.4.2 ”开启 事务 


首先 ， 试 着 显示 表 tb 中 的 数据 。 











执行 结果 
mysql> SELECT * FROM tb; 
2 由 下 + 


| empid | sales | month | 
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Al02 54 5 
Al04 181 4 
Al01 184 4 
A103 EE 5 
Al01 300 县 
Al02 205 6 
Al04 93 5 
Al03 12 6 
A1L07 87 6 
+------- +------- +------- 十 
10 rows in set (0.13 sec) 








所 有 的 数据 都 在 。 
下 面 开 启事 务 。 开 启事 务 时 使 用 START TRANSACTION。 试 着 执行 该 命令 。 


开启 事务 











或 者 可 以 输入 BEGIN 或 BEGIN WORK。 请 确认 执行 结果 中 是 否 像 下 面 这 样 显示 了 Query OK 
消息 。 


mysql> START TRANSACTION; 
Query OK, 0 rows affected (0.07 sec) 






































如 果 未 显示 Query OK， 则 表示 事务 未 启动 ， 操 作 不 会 撤销 。 
下 面试 着 删除 表 tb 中 所 有 的 记录 。 


DELETE FROM tb; 


mysql> DELETE FROM tb; 
Query OK, 10 rows affected (0.13 sec) 














因为 没有 加 上 WHERE 等 条 件 ， 所 以 表 tb 中 的 所 有 记录 都 会 被 删除 。 

此 时 表 的 一 部 分 功能 会 加 锁 (lock )。 因 此 ， 在 其 他 会 话 中 不 能 对 该 表 执行 INSERT 等 操作 。 
假设 这 时 我 们 再 启动 一 个 命令 提示 符 ， 并 在 MySQL 监视 器 上 对 表 tb 执行 TINSERT。 在 这 种 情况 
下 ， 只 有 在 开启 了 事务 的 MySQL 监视 器 中 执行 了 “CoMMIT; ”或 “ROLLDBACK;”， 才 能 进行 其 他 
的 会 话 处 理 。 
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13.4.3 ”确认 表 的 内 容 


一 一 一 一 一 一 














现在 表 tb 中 应 该 没有 记录 了 。 我 们 来 





认 一 下 。 请 在 开启 了 事务 的 MySQL 监视 器 中 对 表 的 内 容 











进行 确认 。( 此 时 如 果 在 其 他 命令 提示 符 下 使 月 











To ) 





日 MySQL 监视 器 执行 SELECT， 就 会 看 到 记录 还 在 表 


SELECT * FROM tb; 



































执行 结果 
mysql> SELECT * FROM tb; 
Empty set (0.00 sec) 
执行 结果 中 显示 了 Empty Set， 它 表示 没有 记录 。 那 么 删除 的 记录 真 的 能 够 复原 吗 ? 真是 令 人 紧张 。 
> ” 回 深 复原 
下 面 来 复原 记录 。 这 时 需要 使 用 ROLLBACK 命令 。 











故 于 日 深 ( 复原 ) 


















































ROLLBACK; 
执行 结果 
mysql> ROLLBACK; 
Query OK, 0 rows affected (0.05 sec) 
这 样 ，DELETE 命令 的 结果 应 该 就 不 会 反映 到 数据 库 中 了 。 试 着 用 SELECT 命令 确认 一 下 结果 。 
执行 结果 
mysql> SELECT * FROM tb; 
+------- +------- +------- + 
empid sales month 
+------- +------- +------- + 
A1l03 LO 4 
A1l02 54 S 
Al04 181 4 
Al01 184 4 
A103 17 5 
A1l01 300 3 
A1l02 205 6 
AlL04 33 5 
A103 12 6 
A1L07 87 6 
+------- +------- +------- + 
10 rows in set (0.00 sec) 
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可 以 看 到 ， 记 录 被 顺利 复原 了 。 终 于 可 以 放心 了 。 

当 执 行 “ROLLBACK; ”了 时， 事务 会 被 关闭 。 下 次 再 练习 时 ,务必 使 用 “START TRANSACTION;” 
开启 事务 。 

在 上 面 的 示例 中 如 果 执 行 “COMMIT ;” 代 替 “RoLLBRACK;”， 删 除 记录 的 结果 就 会 提交 ( 反 
映 ) 到 数据 库 中 ， 所 有 的 记录 都 会 被 永久 删除 。 


本 :于 时 提交 (反映 ) 




















13.5 ”自动 提交 功能 


在 MySQL 中 执行 命令 ， 人 处理 通常 会 直接 提交 。 也 就 是 说 ， 所 有 命令 都 会 自动 CoMMIT。 特 别 
是 在 MySQL 5.5 之 前 默认 使 用 的 MyISAM 中 ， 由 于 没有 事务 这 项 功能 ， 所 有 的 命令 都 会 被 提交 。 

这 种 自动 进行 提交 的 功能 ， 称 为 自动 提交 功能 (auto commit ) ( 见 图 13-4 )。 

在 默认 状态 下 ， 自 动 提交 功能 处 于 开启 状态 。 但 是 ， 当 存储 引擎 为 InnoDB 时 ， 如 果 执 行 了 
START TRANSACTION (或 BEGIN )， 在 执行 COMMIT 命令 之 前 就 不 会 提交 。 多 亏 了 这 个 功能 ， 回 
滚 操作 (ROLLBRACK ) 才 得 以 执行 。 

用 户 也 可 以 强制 将 自动 提交 功能 设置 为 关闭 。 如 果 关 闭 了 自动 提交 功能 ， 即 使 执行 SQL 语句 
也 不 会 自动 提交 ， 必 须 通过 COMMIT 进行 提交 ， 或 者 通过 ROLLBACK 进行 复原 。 


a 
A 


处 理 










































































| 


太 自 动 提交 功能 OFF 


在 
1 1 
1 1 
D> ! ! COMMIT 
1 1 
和 I 
ROLLBACK Wy | 


Ce 


13-4 自动 提交 功能 
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® 13.5.1 关闭 自动 提交 功能 


下 面试 着 关闭 自动 提交 功能 。 
我 们 需要 执行 下 面 的 “SET AUTOCOMMIT=0; ”命令 来 关闭 自动 提交 功能 。 


关闭 自动 提交 功能 














mysql> SET AUTOCOMMIT=0; 
Query OK, 0 rows affected (0.09 sec) 














这 样 就 关闭 了 自动 提交 功能 。 但 是 ，DELETE 命令 总 让 人 感到 有 些 不 安 ， 所 以 这 里 我 们 执行 一 
下 INSERT 命令 ， 选 用 适当 的 值 即 可 。 


INSERT INTO tb VALUES('test',555,555); 


mysql> INSERT INTO tb VALUES ('test' ,555,555) 
Query OK, 1 row affected (0.02 sec) 














试 着 确认 一 下 内 容 。 执 行 “SELECT * FROM tb;” 命 令 就 能 显示 出 INSERT 的 记录 了 。 

















mysql> SELECT * FROM tb; 
+------- +------- +------- 十 
empid sales month 
+------- +------- +------- 十 
A103 L001 4 
A1l02 54 5 
A1l04 181 4 
A1l01 184 4 
Al103 1 5 
Al01 300 二 
Al102 205 6 
Al04 $3 3 
A103 12 6 
A1l07 87 6 
test S53 S55 
+------- +------- +------- 十 
11 rows in set (0.04 sec) 
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那么 ， 回 滚 操 作 能 和 否 执行 成 功 呢 ? 


ROLLBACK; 


这 时 如 果 执 行 “SELECT * FROM tb;”，INSERT 的 记录 就 不 会 显示 出 来 。 
































mysql> SELECT * FROM tb; 
+------- +------- +------- + 
empid sales month 
+------- +------- +------- 十 
Al103 101 4 
A1l02 54 S 
Al104 业 8 焉 4 
Al101 184 4 
A1l03 3 S 
Al101 300 多 
Al102 205 6 
Al104 93 5 
A1l03 12 6 
A1l07 37 6 
和 3 下 二 + 
10 rows in set (0.00 sec) 











可 以 看 到 数据 没有 自动 提交 。 要 想 提交 实际 修改 的 数据 ， 必 须 执行 “coOMMIT;”。 

有 些 人 可 能 认为 如 果 总 是 将 自动 提交 功能 设置 成 关闭 状态 就 可 以 随时 执行 回 深 操 作 了 。 需 要 注 
意 的 是 ， 在 自动 提交 功能 关闭 的 状态 下 ， 如 果 没 有 进行 COMMIT 就 退出 MySQL， 工 作 内 容 就 不 会 
反映 到 数据 库 中 。 








全 13.5.2 启动 已 关闭 的 自动 提交 功能 
下 面 将 介绍 如 何 启动 已 关闭 的 自动 提交 功能 。 请 大 家 按照 以 下 操作 将 自动 提交 功能 恢复 为 原来 
的 启动 状态 。 


SET AUTOCOMMIT=1; 


mysql> SET AUTOCOMMIT=1; 
Query OK, 0 rows affected (0.00 sec) 
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如 何 确认 当前 自动 提交 功能 的 模式 ? 
我 们 可 以 通过 SELECT @@AUTOCOMMIT 确认 当前 自动 提交 功能 的 模式 。 如 果 是 开启 状态 ， 则 显示 为 “1”， 
如 果 是 关闭 状态 ， 则 显示 为 “0”。 























启用 事务 之 后 ， 并 不 是 所 有 操作 都 可 以 通过 回 深 复 原 。 例 如 ， 下 面 这 些 命令 就 会 被 自动 提交 。 





SDROP DATABASE 
SDROP TABLE 
SDROP VIEW 
SALTER TABLE 














即使 开启 了 事务 也 有 不 能 复原 的 情况 ， 这 一 点 需要 注意 。 


本 章 介绍 了 以 下 内 容 。 


@ 什 么 是 存储 引擎 ， 都 有 哪些 存储 引擎 
@ 什 么 是 事务 

@ 开 启事 务 的 方法 

@ 提 交 和 回 滚 的 方法 

@ 什 么 是 自动 提交 功能 





能 够 自由 选择 存储 引擎 也 是 MySQL 的 魅力 之 一 。 我 们 可 以 使 用 SHOW ENGINES NG 显示 当 
前 可 用 的 存储 引擎 。 由 此 至 少 可 以 知道 ， 在 本 书 使 用 的 环境 中 支持 事务 的 存储 引擎 只 有 InnoDB。 











> 自我 检查 
下 面 检查 一 下 本 章 学 习 的 内 容 是 否 全 部 理解 并 掌握 了 。 




















口 理解 了 存储 引擎 的 相关 内 容 
口 能 够 开启 事务 
口 能 够 显 式 地 进行 提交 或 者 回 滚 








口 知道 哪些 命令 无 法 回 滚 


二 练习 题 


口 了 解 了 自动 提交 功能 并 能 够 对 其 进行 设置 


13.6 


Ek 
雯 
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问题 1 














































































































存储 引擎 为 InnoDB 的 表 t_tran 如 下 所 示 。 开 启事 务 并 执行 “UPDATE t tran SET a=777;” 后 

在 没有 提交 的 情况 下 直接 退出 MySQL 监视 器 。 之 后 ， 再 次 启动 MySQL 监视 器 ， 确 认 数据 的 变化 情况 。 
> 表 t tran 

列 名 外 

列 的 数据 类 型 INT 

列 的 内 容 100 

HINT] . 和 

| 这 只 是 一 个 实验 而 已 。 确 认 表 t_tran 的 存储 引擎 为 InnoDB 后 再 执行 相关 操作 ( 一 13.2.1 节 ) 

[Ie 



































执行 下 面 的 命令 。 
确认 表 t_tran 的 内 容 (a 为 100 )。 











mysql> SELECT * FROM 七 tran; 


1 row in set (0.00 sec) 
@ 开启 事务 。 


mysql> START TRANSACTION; 


Query OK, 0 rows affected (0.00 sec) 
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@) 执行 UPDATE 并 确认 执行 后 的 数据 (a 被 修改 为 777 )。 








mysql> UPDATE 七 tran SET a=777; 
Query OK, 1 row affected (0.06 sec) 
Rews matened:m nenanged wanings no 


mysql> SELECT * FROM t tran; 


1 row in set (0.00 sec) 











由 通过 exit 命令 退出 MySQL 监视 右 。 
@) 启动 MySQL 监视 器 并 确认 表 t tran 的 内 容 (a 恢复 为 原来 的 100。 把 a 修改 为 777 的 处理 














作废 )。 


mysql> SELECT * FROM t tran; 


1 row in set (0.00 sec) 





如 果 没 有 提交 事务 就 退出 MySQL 监视 顺 ， 修 改 的 内 容 就 会 作废 。 也 就 是 说 ， 如 果 在 未 提交 





务 的 情况 下 关闭 连接 ， 事 务 将 自动 回 深 。 

















= 
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在 MySOL 监视 器 这 种 基于 CUI ( 仅 限 于 字符 ) 的 环境 中 执 











行 复杂 的 SQL 语句 ， 或 者 插入 所 有 数据 是 有 一 定局 限 性 的 。 





本 章 就 来 总 结 一 下 使 用 文件 处 理 数 据 的 方法 。 





T 





























14.1 从 文本 文件 中 读 取 数据 ( 导入 ) 


当 把 大 量 数据 输入 到 表 中 时 ， 如 果 打 开 MySQL 监视 器 用 键盘 手动 输入 所 有 数据 ， 就 会 耗费 非 


常 多 的 时 间 和 精力 。 











在 需要 输入 成 千 上 万 条 数据 的 情况 下 ， 我 们 可 以 使 用 CSV (Comma Separated Values， 逗 号 分 
隔 值 ) 格式 的 文本 文件 进行 输入 ， 这 种 读 取 文件 的 方式 称 为 导 人 (import )。 


Ce 14.1.1 CSV 文 件 


正如 逗号 分 隔 值 这 个 名 字 所 表达 的 那样 ， 在 CSV 文件 中 ， 数 据 是 用 逗号 隔 开 的 ， 文 件 内 容 仅 


包含 文本 ( 见 图 14-1 )。 





文件 的 内 容 
只 包括 文本 











广 -一 一 一 6 N551, 佐佐木 ,37 @ 一 一 一 一 换行 




















N552,， 伊 芯 ,41 


N553， 齐 采 ，31 











N554, 井上 ,43 





N555, 阿 倍 ，31 
































图 14-1 


CSV 格式 的 文件 
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每 条 记录 都 通过 换行 符 用 单独 的 一 行 表示 ( 见 图 14-2 )。 
CSV 文 件 读 取 表 
empid ，name ' age 
N551, 佐 佐 木 ,37 一 一 一 一 一 一 一 一 一 一 | N551 ' 佐佐木 '， 37 
N552, 伊 芯 ,41 甩 | N552 ， 伊 蕨 ， 41 
N553, 齐 苹 ,31 也 | N553 ; 章节 31 
N554, 井 上 ,43 “一 人 儿 ， N554 ， EE 43 
N555, 阿 倍 ,31 胞 | N555 : 阿 倍 31 
图 14-2 导入 
14.1.2 ”导入 和 导出 的 准备 


@ 一 一 二 人 一 
如 果 没 有 根据 MySQL 的 版 本 更 新 配置 文件 ， 就 无 法 导入 或 导出 文件 。MAMP 3.3.1 中 包含 的 











有 相应 权限 的 情况 下 对 服务 右上 的 文件 进行 读 取 和 写 入 ， 














MySQL 也 需要 更 新 。 当 然 ， 如 果 在 没有 所 
就 可 能 会 出 现 安全 方面 的 问题 。 


本 书 作为 一 本 入 门 书 ， 没 有 对 MySQL 的 用 户 权限 进行 说 明 ， 所 以 我 们 选择 月 





最 简单 的 方法 导 





入 和 导出 文件 。 请 按照 2.4.2 节 所 述 ， 从 C:\MAMP\conf\mysql 这 个 路 径 打 开 MySQL 的 设置 文件 
my.ini。 然 后 在 第 41 行 附 近 添 加 下 面 的 红字 部 分 。 


代码 清单 14-1 修改 my.ini 





# The MySQL server 
[mysqld] 


3306 

mysql 
skip-external-locking 
16M 


port 


socket 


key_ buffer size 


max allowed packet = 1M 
table open cache = 64 

sort buffer size = 512K 

net buffer length = 8K 

reagd buffer Size = 256K 

read rnd buffer size = 512K 
myisam sort buffer size = 8M 


basedir 
datadir 


C:/MAMP/bin/mysql/ 
C:/MAMP/db/mysqgl/ 
character-set-server=utf8 


Sectre-Tile=DEi = ?7 
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保存 文件 后 ， 请 在 MAMP 的 启动 画面 中 重新 启动 服务 器 。 然 后 ， 启 动 MySQL 监视 器 ， 执 行 
下 面 的 命令 。 如 果 列 上 没有 显示 任何 内 容 ， 则 表示 成 功 。 





























执行 结果 





mysql> SELECT Q@Q@global.secure _ file priv; 





1 row in set (0.00 sec) 














@@global 是 用 于 引用 MySQL 系统 全 局 变量 的 关键 字 。 连 接 到 这 个 MySQL 服务 器 的 所 有 。 
户 端 都 会 引用 系统 全 局 变量 的 值 ，secure file priv 用 于 指定 允许 导入 、 导 出 文件 的 路 径 
果 指 定 为 空 字符 串 "" ， 那 么 任何 路 径 都 可 以 导入 或 导出 文件 。 如 果 指 定 为 NULL， ee 
出 文件 。 








6 14.1.3 ”导入 文件 


我 们 可 以 使 用 LOAD DATA INEILE 命令 从 文件 导入 数据 ， 有 具体 语句 如 下 所 示 。 
从 文件 中 读 取 数据 





LOAD DATA INFILE ' 文件 名 ' INTO TABLE 表 名 选项 的 描述 ; 

除了 CSYV 格式 的 文件 以 外 ， 不 用 逗号 分 隔 的 文本 文件 也 能 被 读 取 。 我 们 可 以 指定 读 取 的 数据 
的 格式 ， 比 如 指定 ed 换行 符 ， 以 及 从 第 几 行 开始 读 取 等 。 

在 这 种 情况 下 ,“ 选 项 的 描述 ”部 分 需要 按照 下 面 的 格式 编写 。 


LOAD DATA INFILE 命令 中 指定 数据 格式 的 选项 








FIELDS TERMINATED BY 分 隔 符 ( 默认 是 '\t' : Tab ) 
; LINES TERMINATED BY 换行 符 (默认 是 '\n' : 换行 ) 
: IGNORE 最 开始 跳 过 的 行 LINES ( 默认 是 0 ) 








前 一 节 介 绍 的 文件 (代码 清单 14-2 ) 和 表 tbl 的 结构 相同 。 这 次 ， 我 们 试 着 将 这 个 文件 命名 为 
tcsv， 并 将 其 导入 到 结构 与 表 tbl 相同 的 表 tb1K 中 。 
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代码 清单 14-2 tcsv 





N551, 佐佐木 ,37 
N552，, 伊 苹 ,41 
N553， 齐 蕨 ，31 
N554, 井上 ,43 
N555, 阿 倍 ，31 











P 使 用 LOAD DATA INFILE 插入 的 文本 的 字符 编码 


在 本 书 中 ，MySQL 监视 器 客户 端的 字符 编码 为 GBK， 服 务 器 端的 字符 编码 为 UTF-8。 那 么 ， 
使 用 LOAD DATA INFILE 命令 插入 的 文本 的 字符 编码 ， 应 该 是 GBK 还 是 UTF-8 呢 ? 

实际 上 ， 在 单纯 使 用 LOAD DATA INFILE 命令 输入 数据 的 情况 下 ， 一 般 会 使 用 数据 库 的 字符 
编码 。 也 就 是 说 ， 此 次 读 取 的 文本 文件 t.csv 需要 使 用 字符 编码 UTF-8。 

因为 下 载 的 数据 样本 tcsy 的 字符 编码 是 UTF-8， 所 以 可 以 直接 导入 该 样本 。 另 外 ，MySQL 也 可 
以 读 取 字 符 编 码 为 GBK 或 GBK 的 子 集 GB 2312 的 CSV 文件 。 相 关 的 操作 方法 会 在 后 面 进行 介绍 。 


P 导入 文件 


CSV 文件 tcsv 使 用 的 分 隔 符 是 “,”， 所 以 我 们 只 要 通过 FIELDS TERMINRATED BY '，' 进 
行 指定 即 可 。 另 外 ， 当 指定 保存 CSV 文件 的 文件 夹 路 径 时 ， 即 使 在 Windows 的 情况 下 也 不 要 使 用 
“\”， 而 要 使 用 “/”。( 例如 : Ci\data\t.csv 一 C:/data/t.csv ) 

下 面 来 实际 操作 一 下 吧 。 假设 上 述 CSV 文件 tcsv 放 在 了 Ci\data 文件 夹 中 ,我们 试 着 将 文件 导入 到 
和 表 tbl 内 容 相同 的 表 tb1K 中 。(t.csv 放 在 了 下 载 文件 的 MySQL _Book\chapte\CHAPTER14 文件 夹 中 。 ) 


PP CSV 文件 tcsv 


N551,， 佐佐木 ,37 
N552，, 伊 芯 ,41 
N553， 齐 蕨 ，31 
N554, 井上 ,43 
N555, 阿 倍 ,31 









































一 一 一 一 一 一 一 导入 该 文件 




















> 执行 前 ( 执行 导入 的 表 tb1K ) 





empid name age 
A101 佐 爱 40 
A102 高 桥 28 
A103 中 川 20 
A104 渡 边 23 
A105 泽 35 














读 入 到 该 表 
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> 执行 后 

empid name age 
A101 佐藤 40 
A102 高 桥 28 
A103 中 川 20 
A104 渡 边 23 
A105 西 泽 35 
N551 佐佐木 | 37 
N552 伊藤 41 
N55S 齐 茧 31 一 一 一 一 导入 完成 
N554 井上 43 
N555 ”| 阿 倍 31 



















































































操作 方法 





执行 下 面 的 命令 。 

















LOAD DATA INFILE 'c:/data/t.csv' INTO TABLE tblK FIELDS 
TERMINATED BY ','; 





mysql> LOAD DATA INFILE 'c:/data/t.csv' INTO TABLE tblK FIELDS TERMINATED BY ','; 
Query OK, 5 rows affected (0.09 sec) 
Records: 5 Deleted: 0 Skipped: 0 Warnings: 0 











请 使 用 “SELECT * FROM tblK;” 命 令 确 认 表 tblK 的 内 容 。 可 以 看 到 新 添加 了 5 条 记录 。 








执行 结果 

mysql> SELECT * FROM tblK; 
再 和 由 汪汪 和 十 
| empiqd | name | age | 
+------- +-------- +------ + 
| A101 | 佐 苹 | 40 | 
| A102 | 高 李 | 28 | 
| A103 | 中 川 | 20 | 
| A104 | 渡 边 | 23 | 
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| A105 | 西 泽 | 35 | 
| N551 | 佐佐木 | 37 | 
| N552 | 伊 芯 | 41 | 
| N553 | 齐 藤 | 31 | 
| N554 | 井上 | 43 | 
| N555 | 阿 倍 | 31 | 
+------- +-------- +------ + 
10 rows in set (0.00 sec) 

















即使 数据 库 的 种 类 不 同 ， 只 要 文件 是 CSV 格式 就 可 以 执行 导入 。 可 以 说 这 是 操作 数据 库 必 不 
可 少 的 一 项 技术 。 





导入 GB 2312 的 文本 文件 

使 用 LOAD DATA INFILE 命令 将 数据 输入 到 表 中 时 使 用 的 是 数据 库 端的 字符 编码 。 因 为 本 书 以 数据 库 端 
的 字符 编码 是 UTF-8 为 前 提 进 行 介绍 ， 所 以 这 次 我 们 导入 了 字符 编码 为 UTF-8 的 CSV 文件 t.csvo。 

如 果 要 导入 简体 中 文 GB 2312 的 CSV 文件 ， 就 需要 在 LOAD DATA INFILE 的 选项 中 加 上 CHARACTER 
SET GB2312 来 执行 。 下 面 的 命令 用 于 导入 字符 编码 为 简体 中 文 GB 2312 的 CSV 文件 t2.csv。 































































































LOAD DATA INFILE 'c:/data/t2.csv' INTO TABLE tblK CHARACTER 


SET GB2312 FIELDS TERMINATED BY ',' ，} 





dl4 将 数据 写 入 文本 文件 ( 导出 ) 
与 导入 相反 ， 我 们 可 以 将 表 中 的 数据 提取 到 CSV 文件 等 文本 文件 中 ( 见 图 14-3 )。 这 种 把 数据 
提取 到 文件 中 的 操作 称 为 导出 (export )。 


表 CY CSV 文 件 
empid ， | 


N551 ， 佐 佐 木 ， 37 N551, 佐 佐 木 ,37 
N552 ， 伊 芯 ， 41 入 N552, 伊 芯 ,41 
N553 : 齐 蕨 ; 31 区 N553, 齐 芯 ,31 
N554 ， 井上 ， 43 N554, 井 上 ,43 
N555 ， | N555, 阿 倍 ,31 














































































图 14-3 导出 
导出 的 文件 可 以 在 其 他 的 数据 库 和 系统 中 使 用 ， 也 可 以 在 紧急 情况 下 作为 备份 使 用 。 
我 们 可 以 通过 如 下 命令 执行 导出 操作 。 
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将 数据 写 入 到 文本 文件 


eeeeeeseeeeeeeeeeooeseseooooeeooooooeoeooooooooooooeooooosoooooeoeoooooooooooooeoeoooooooeooooeooooooooooeooooooooooooeeooooooeoeoooooeeooooeeeooe 


eeeeeeeeeeeeeeeoooooooeoeoooeooooooooooooooooooooooeooooooooooooeoooooooooooooooooooooooooooooooooooooooooooooooooeooooooooooooooeooooeoooe 


参数 “选项 的 描述 ”用 于 指定 导出 的 文本 文件 的 格式 。 具 体 的 描述 方法 与 导入 时 完全 相同 ， 大 
家 可 以 参考 14.1.3 节 的 内 容 。 


S 14.1.5 ”导出 文件 


下 面 来 试 着 导出 文件 。 在 C:\data 文件 夹 中 ,创建 CSV 格式 的 文本 文件 out.csv 用 于 导出 表 tb1 
中 的 数据 。 


P> 执行 内 容 


> 执行 前 ( 表 tb1 ) 





empid name age 














@ 一 一 一 一 导出 该 表 的 数据 








BA 


戎 执行 后 ( CSV 文件 out.csv ) 











导出 的 CSV 数据 











操作 方法 
执行 下 面 的 命令 。 


: SELECT * INTO OUTFILE 'C:/data/out.csv' FIELDS TERMINRTED 
: BY ',' FROM tbl; 
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执行 结果 





mysql> SELECT * INTO OUTFILE 'C:/data/out.csv' FIELDS TERMINRATED BY ',' FROM tbl: 
Query OK, 5 rows affected (0.06 sec) 














于 是 ,一 个 名 为 out.csv 的 CSV 文件 就 在 C:\data 文件 夹 中 创建 完成 了 。 我 们 可 以 在 文本 编辑 器 
中 确认 导出 文件 的 内 容 ( 见 图 14-4 )。 


eee EE 
上 文件 (站 ”编辑 (E) ”搜索 ($) 过 看 M 比较 (9 宏 (M) 工具 (窗口 IW) 者 助 (H) | 
DvYBrvnH| 和 DIX 人 DIO QPNDICU7TI ”IIa” 
时 out.csvy x 
A101, 伴 芯 ,40 
A102, 高 桥 , 28J 
A103, 中 川 , 20 
A104, 渡 边 , 23 











A105, 西 泽 , 35 
2 








Y 




















Text 行 1 列 1 U+0041 (0x41) 仅 LF(UNIX) 0 字符 0 单词 0/6 行 12/ 





14-4 ”导出 的 文件 
这 里 需要 注意 的 是 导出 的 CSV 文件 的 字符 编码 。 因 为 CSV 文件 是 通过 MySQL 的 默认 字符 编 
码 导出 的 ， 所 以 在 本 书 环境 下 CSV 文件 的 字符 编码 是 UTF-8。 
文件 out.csv 的 数据 可 以 通过 14.1.3 节 的 LOAD DATA INFILE 命令 导入 。 

















14.2 ”从 文件 中 读 取 并 执行 SQL 命令 


5 14.2.1 通过 MySQL 监视 器 执行 编写 在 文件 中 的 SQL 语句 


当 执行 复杂 且 宛 长 的 SQL 语句 时 ， 如 果 每 次 都 在 MySQL 监视 需 上 编写 就 会 很 豚 烦 。 在 执行 复 
杂 的 SQL 语句 的 情况 下 ， 我 们 可 以 将 其 创建 为 文本 文件 ， 然 后 执行 保存 的 文件 ( 见 图 14-5 )。 这 种 
方法 可 以 进一步 提高 工作 效率 ， 还 可 以 显示 创建 成 文本 文件 的 SQL 语句 ， 并 将 其 复制 、 粘 由 到 
MySQL 监视 右上 执行 。 

如 果 将 SQL 语句 保存 为 文本 文件 ， 就 可 以 反复 使 用 它 ， 还 可 以 轻松 对 其 进行 改善 。 
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| 创建 文本 文件 ] 


v 


INSERT INTO TABLE ... 
BDAMNENTABLE SET ... 
SELECT COLUMN FROM ... 











汇总 SQL 语 句 








INSERT INTO TABLE ... 
UPDATE TABLE SET ... 
SELECT COLUMN FROM ... 


表 
列 1 ， 列 2 ， 列 3 


A3 
Ca 
A5 





























14-5 读 取 SQL 文件 
在 记事 本 等 文本 编辑 器 中 事先 准备 好 SQL 语句 ， 然 后 在 MySQL 监视 器 上 执行 SOURCE 命令 。 








执行 包含 SQL 语句 的 文本 文件 


如 果 执 行 的 SQL 语句 中 包含 中 文 ， 就 需要 注意 作为 参数 的 文本 文件 的 字符 编码 了 。 使 用 
SOURCE 命令 执行 SQL 语句 的 方式 与 在 MySQL 监视 器 上 执行 的 方式 相同 ， 因 此 在 本 书 使 用 的 环境 
中 文本 文件 的 字符 编码 必须 保存 为 GBK 或 GBK 的 子 集 GB 2312。 

下 面 来 实际 操作 一 下 。 试 着 在 C:\data 文件 夹 中 创建 包含 “use dbl” “SELECT * FROM 
tb;” 和 “SELECT * FROM tbl;” 这 3 行 语句 的 文本 文件 sql.txt， 读 取 并 执行 该 文本 文件 。 



























































P 执行 内 容 


> ( 执行 前 ) 文本 文件 sql.txt 


use dbl 
SELECT * FROM tb; @ 一 一 一 读 取 并 执行 该 文本 文件 



































SELECT * FROM tb1l; 


D4 


指定 数据 库 db1 ee 一 一 一 一 一 一 use dbl (执行 sqltxt ) 
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D4 


> ( 执行 后 ) 显示 表 tb 















































empid sales month 
A103 101 4 
A102 54 5 
A104 181 4 
A101 184 4 
@ -SELECT * FROM tb; 
A103 17 5 
A101 300 5 
A102 205 6 
A104 93 5 
A103 己 6 
A107 87 6 

















PP (执行 后 ) 显示 表 tb1 







































































empid name age 
A101 佐 世 40 
A102 高 桥 28 
全 SELECT * FROM tbl; 
A103 中 川 20 
A104 渡 边 23 
A105 泽 35 
操作 方法 





人 创建 代码 清单 14-3 的 文本 文件 sql.txt。 
人) 执行 下 面 的 命令 。 








代码 清单 14-3 sql:txt 





use dbl 
SELECT * FROM tb; 
SELECT * FROM tbl; 
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mysql> SOURCE C:/data/sql.txt 
Database changed 














+------- +------- +------- 十 
empid sales month 
站 本 本 + 
A1l03 101 4 
Al102 54 S 
Al104 18 4 
Al01 184 4 
Al103 17 号 
Al01 300 5 
Al102 205 6 
Al104 93 5 
A1l03 Ta 6 
Al107 87 6 
+------- +------- +------- 十 


10 rows in set (0.09 sec) 



































和 re de 
empid name age 
i 下 所 人 + 
A1l01 佐 藻 40 
A102 高 村 28 
RAR103 中 川 20 
RA104 渡 边 23 
RA105 西 泽 35 
ed a nt + 


5 rows in set (0.02 sec) 











因为 SOURCE 不 是 SQL 命令 ， 所 以 不 需要 在 行 的 最 后 添加 分 隔 符 “;”。 


将 常用 表 的 内 容 保存 到 文件 中 

使 用 MySQL 监视 器 创建 “需要 多 次 创建 的 表 ” 或 者 “复杂 的 表 ” 效 率 会 很 低 。 在 这 种 情况 下 ， 建 议 在 文本 
中 编写 处 理 步 骤 ， 然 后 执行 SOURCE 命令 来 创建 表 。 

创建 好 文本 文件 后 ， 就 可 以 根据 需要 反复 地 使 用 它 ， 并 且 可 以 不 断 地 对 其 进行 改善 。 最 重要 的 是 ， 与 直接 执 
行 不 同 ， 我 们 可 以 静 下 心 来 编写 ， 由 此 也 可 以 减少 输入 方面 的 错误 。 

例如 ， 代 码 清单 14-4 是 创建 2.5.2 节 员 工 信 息 表 tb1 的 SQL 语句 。 

























































































































































































302 | 第 14 章 使 用 文件 进行 交互 


代码 清单 14-4 tb1_make.txt 





DROP 


INSER 
INSER 


INSER 





INT 





IN 





INSERT 


IN 


INT 








IN 





O 


O 
O 
oO 
O 


tL 
DL 
til 
tbl 
tbl 


TABLE IF EXISTS tbl; 
CREATE TABLE tbl 
INSERT 








VALUES ("A101"," 佐藤 "， 
VALUES ("A102"," 高 桥 "， 
VALUES ("A103"," 中川" 
VALUES ("A104"," 渡 边 "， 
VALUES ("A105"," 西 泽 "， 























(empid VARCHAR(10),name VARCHAR (10) ,age INT); 


40); 














字符 编码 改 为 GB 2312。 


非常 方便 。 








使 用 文本 编辑 器 创建 好 文件 后 ， 将 其 保存 在 data 文 








件 夹 

















Pp。 在 本 书 使 用 的 环境 下 ， 注 意 不 要 忘记 将 文件 的 








启动 MySQL 监视 器 选择 数据 库 ， 然 
































后 执行 下 面 的 脚本 ， 就 可 以 随时 创建 相同 的 表 了 ， 











SOURCE C:/data/tb1 make.txt 





3 外 ， 因 











14.2.2 





通 


use db1 
SELECT * FROM tb; 
SELECT * FROM tbl; 


添加 - e 选项 ， 然 后 将 后 面 


AM 


儿 





为 tb1_make.txt 的 第 1 行 是 IF EXISTS， 所 以 在 有 表 的 情况 下 会 执行 DROP 命令 ， 在 没有 表 的 情 
况 下 则 不 执行 DROP 命令 。 








之 前 执行 SQL 命令 的 时 候 会 先 启 动 MySQL 监视 器 并 在 mysql> 之 后 输入 SQL 语句 执行 。 实 
际 上 ， 我 们 也 可 以 不 启动 MySQL 监视 器 ， 直 接 通 过 命令 提示 符 来 执行 SQL 语句 。 
也 就 是 说 ， 即 使 不 启动 MySQL 监视 器 ， 也 可 以 执行 包含 SQL 语句 的 文本 文件 。 这 里 我 们 试 着 


使 用 SOURCE 命令 执行 代码 清单 14-3 。 



































的 命令 用 " " 括 起 来 。 注 意 要 用 "" ( 双 引 号 ) 将 命令 括 起 来 ， 而 不 





是 用 ' ' ( 单 引 号 )。 另 外 ,请 在 -p 密码 、-e 和 "MySQL 监视 器 的 命令 " 之 间 加 上 半角 空格 。 
如 果 不 需 要 指定 数据 库 名 ， 则 可 以 省 略 “ 数 据 库 名 ”部 分 。 因 为 在 sqltxt 中 有 use dbl 的 描 
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述 ， 所 以 可 以 省 略 “数据 库 名 ”。 
我 们 来 实际 操作 一 下 。 文 本 文件 sql.txt 中 编写 了 数据 库 dbl 相关 的 SQL 语句 ， 试 着 通过 
mysql 命令 执行 保存 在 C:\data 文件 夹 中 的 该 文本 文件 。 


操作 方法 
















































































执行 结果 
C:\Users\nisizawa>mysql dbl -u root -proot -e "SOURCE C:/data/sql.txt" 
Warning: Using a password on the command line interface can be insecure. 
下 = +------- 下 = 二 + 
empid sales month 
+------- +------- 可 全 这 生生 + 
Al103 二 总 册 4 
AL02 54 S 
Al04 181 4 
Al01 184 4 
A1l03 13 各 
AL01 300 5 
Al02 205 6 
Al04 93 s 
AL03 12 6 
A1l07 87 6 
+------- +------- +------- + 
+------- +------ +------ 十 
empid name age 
+------- +------ +------ + 
A1l01 佐藤 40 
A1l02 高 村 28 
A103 中 川 20 
Al04 渡 边 23 
ARA105 西 泽 35 
+------- +------ +------ 十 











即使 不 启动 MySQL 监视 器 ， 也 可 以 通过 命令 提示 符 直 接 执 行文 本 文件 中 编写 的 SQL 命令 。 
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在 批 处 理 文件 中 记录 SQL 命令 

如 果 将 可 以 通过 命令 提示 符 执行 的 命令 汇总 为 文本 文件 ， 编 写 在 一 个 扩展 名 为 “ .bat” 的 批 处 理 文件 中 ， 
使 用 起 来 就 会 非常 方便 。 如 果 提 前 设置 好 MySQL 的 路 径 ( 一 2.3 节 )， 就 可 以 从 任何 地 方 对 MySQL 进行 操作 。 
举例 来 说 ， 如 果 将 14.1.5 节 中 记述 的 文件 操作 作为 批 处 理 文件 保存 在 桌面 上 ， 就 可 以 随时 从 桌面 上 导出 表 。 
将 数据 库 db1 的 表 tb1 写 入 Cdatavout2.csv ( CSV 文件 ) 的 批 处 理 文件 out_file.bat 如 下 所 示 。 





































































































@ out file.bat 的 内 容 ( 必须 用 一 行 描述 ) 























mysql dbl -u root -proot -e "SELECT * INTO OUTFILE 'C:/data/out2.csv' 


FIELDS TERMINATED BY ',' FROM tbl" 





14.3 将 SQL 的 执行 结果 保存 到 文件 中 


在 MySQL 监视 莫 上 输入 的 命令 会 显示 在 屏幕 上 ， 执 行 结果 也 会 显示 在 屏幕 上 。 用 户 可 以 通过 
查看 屏幕 上 显示 的 执行 结果 来 获取 信息 ， 也 可 以 将 这 些 执行 结果 作为 数据 使 用 。 

那么 ， 如 何 将 SQL 的 执行 结果 保存 到 文件 中 呢 ? 这 里 我 们 将 介绍 “在 MySQL 监视 器 上 执行 
tee 命令 ”和 “使 用 重 定向 将 结果 输出 到 标准 输出 ”这 两 种 方法 。 






































人 14.3. 通过 重 定向 将 SQL 语句 的 执行 结果 输出 到 文本 文件 中 

我 们 可 以 在 计算 机 上 输入 一 些 数据 并 让 计算 机 输出 处 理 结果 。 这 时 通常 会 使 用 键盘 进行 “ 输 
入 ”， 然 后 把 结果 “输出 ”到 显示 器 上 。 像 键盘 这 种 一 开始 就 配备 好 的 输入 设备 称 为 “标准 输入 ”， 
像 显 示 器 这 种 一 开始 就 配备 好 的 输出 设备 称 为 “标准 输出 ”。 

“标准 输入 ”和 “标准 输出 ”可 以 更 改 。 这 个 更 改 操 作 称 为 重 定向 (redirect )，Windows 和 
Linux 等 操作 系统 都 具有 这 项 功能 。 当 进行 重 定向 操作 ， 也 就 是 更 改 输入 、 输 出 目标 时 ， 需 要 使 用 
“>” 等 符号 。 





























P 通过 命令 提示 符 进 行 重 定向 
例如 ， 在 Windows 命令 提示 符 下 输入 dir (如 果 是 Linux 终端 则 输入 1s )， 将 会 显示 文件 和 文 
件 夹 的 信息 。 





和 是 电 示 的 内 容 会 根据 环境 发 生 改 变 
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2018/06/24 
2018/06/24 
2017/08/14 
2017/09/16 
2018/04/11 


2018/02/12 
2018/02/12 
2018/02/12 
2018/04/11 
3 个 文件 























dl 
Ls 
14 
16 
20 


09: 
O09 
09: 
203 


C:\Users\nisizawa 的 


二 了 
17 


:45 
16 
222 


47 
47 
47 
22 


C:\Users\nisizawa>dir 
驱动 器 C 中 的 卷 是 System 
卷 的 序列 号 是 38E8-F275 





























录 
<DIR> 
<DIR> 
<DIR> 
<DIR> 
<DIR> 

1R> 

IR> 

<DIR> 
<DIR> 
1,071 字 节 





18 个 目录 80,300,621,824 可 用 字 节 


5456G 
-Oracle jre Usage 
.VirtualBox 


Saved Games 
Searches 
Videos 
VirtualBox VMs 








也 就 是 说 ，dir 的 结果 输出 到 了 显示 器 这 个 标准 输出 中 。 试 着 使 用 重 定向 将 这 个 执行 结果 保存 在 


非 标准 输出 的 abc.txt 文件 


| 执行 结果 














P。 请 按照 以 下 步骤 进行 操作 。( 在 Linux 的 情况 下 是 ls > abc.txto) 








C:\Users\nisizawa>dir > abc.txt 








这 样 一 来 ， dir 的 执行 结果 就 写 入 了 文本 文件 abc.txt 中 。 之 所 以 会 写 入 文本 文件 abc.txt 中 

















是 因为 加 上 了 > abc.txt。 

保存 了 输出 结果 的 文件 abc.txt 会 保存 在 执行 air 命令 的 当前 路 径 中 。 拿 上 面 的 示例 来 说 ， 就 
是 保存 在 C 盘 的 \Usersnisizawa 文件 夹 (目录 ) 中 。 请 在 文本 编辑 器 或 命令 提示 符 下 执行 Eype 
abc .txt 来 确认 文件 内 容 。 应 该 和 执行 dir 的 时 候 显 示 的 内 容 相 同 。 


> 通过 mysql 命令 使 用 重 定向 
下 面试 着 在 MySQL 中 使 用 重 定向 功能 。 假 设 在 启动 MySQL 监视 咒 的 时 候 执 行 了 以 下 命令 。 





mysql -u root -proot 















































试 着 使 用 重 定向 把 提取 结果 输入 到 文件 中 。 请 按照 下 面 的 方式 启动 MySQL 监视 器 。 刚 开始 会 
显示 “Warning: Using a bassword...” 的 警告 ， 大 家 不 必 在 意 。 


mysql -u root -proot > log.txt 
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> 








如 果 按 照 这 种 方式 启动 MySQL 监视 器 ，SQL 语句 的 执行 结果 将 不 会 显示 在 屏幕 上 ， 而 是 会 加 
出 到 重 定向 指定 的 log.txt 中 。 注 意 ， 不 要 因为 没有 显示 任何 结果 而 胡乱 操作 。 

下 面 让 我 们 静 下 心 来 慢 慢 操作 。 先 指定 要 使 用 的 数据 库 ， 然 后 使 用 SELECT 命令 显示 表 tbl 的 
内 容 ， 最 后 通过 exit 退出 MySQL 监视 器 。 执 行 结果 同样 不 会 显示 出 来 。 





= 















































use dbl 
SELECT * FROM tbl; 


exit 


C:\Users\nisizawa>mysql -u root -proot > log.txt 








Warning: Using a password on the command line interface can be insecure. 
use dbl 

SELECT * FROM tbl: 

exit 

















回 到 原来 的 命令 提示 符 界 面 了 吗 ? 屏幕 上 没有 显示 任何 内 容 ， 确 实 有 点 不 方便 。 

那么 我 们 就 来 确认 一 下 log.txt 的 内 容 。log.txt 应 该 保存 在 执行 命令 的 当前 路 径 中 ( 本 例 为 
Ci\Users\nisizawa )。 使 用 记事 本 等 文本 编辑 器 打开 log.txt， 或 者 在 命令 提示 符 下 输入 type 1l1og. 
txt。 你 会 发 现 ， 以 前 显示 在 MySQL 监视 右上 的 内 容 变 成 了 文本 文件 。 
































执行 结果 log.txt 的 内 容 











C:\Users\nisizawa>type log.txt 
empid name age 

A1l01 佐 腾 40 

RA102 高 桥 28 

A1l03 中 川 20 

Al04 渡 边 23 

A1l05 泽 35 


















































这 样 ,，“SELECT * FROM tbl;” 的 结果 应 该 就 能 显示 出 来 了 。 该 文件 只 是 一 个 文本 文件 ， 
因此 可 以 在 其 他 系统 和 应 用 软件 中 自由 使 用 。 

如 果 需 要 让 结果 显示 在 屏幕 上 ， 则 可 以 结合 14.2.2 节 中 介绍 的 方法 使 用 。 从 命令 提示 符 上 执行 
SQL 语句 文本 文件 sqLtxt， 然 后 使 用 重 定 向 将 结果 直接 写 人 文本 文件 log.txt 中 。 





mysql -u root -proot -e "SOURCE c:/data/sql.txt" >log.txt 





sql.txt 是 在 14.2.1 节 创 建 的 包含 以 下 内 容 的 文本 文件 。 
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sql.txt 的 内 容 





use dbl 
SELECT * FROM tb; 
SELECT * FROM tbl; 














使 用 重 定向 输入 SQL 语句 ， 并 通过 重 定向 将 结果 输出 到 文件 
面 介绍 使 用 重 定向 进行 输入 、 输 出 的 示例 。 
E 定 向 输入 并 执行 文本 文件 的 SQL 语句 ( sql.txt )， 


Ji20 


下 
定向 输入 文件 时 需要 使 用 “<”"。 这 样 就 可 以 使 




















I 















































使 用 重 


然后 再 通过 重 定 向 将 结果 输出 到 文本 文件 ( log2.txt )。 
mysql -u root -proot < C:\ata\sql.txt > C: \data\log2.txt 
这 种 方法 的 情况 下 ， 就 算 SQL 语句 报错 ， 我 们 也 只 能 通过 查看 


即使 看 不 到 过 程 也 没有 问题 。 但 是 ， 在 使 
出 的 文本 文件 得 知 。 


Ce 14.3.2 ”使 用 tee 命令 将 SQL 语句 的 执行 结果 保存 到 文件 中 
将 结果 写 和 人 文件。 如 果 在 


在 MySQL 监视 器 上 使 用 tee 命令 ,可 以 与 上 一 节 的 重 定向 一 术 
MySQL 监视 器 上 按照 如 下 方式 执行 tee 命令 ， 执 行 结果 就 会 保存 在 指定 的 文件 中 。 





































































































将 执行 结果 保存 到 文件 中 


PP 将 执行 结果 保存 到 文件 中 
当 把 执行 结果 导出 到 文件 log3.txt 时 ， 假 设 在 MySQL 监视 器 的 状态 下 执行 了 如 下 操作 。 


在 
tee log3.txt 


| 执行 结果 


mysql> tee log3.txt 
Logging to file 'log3.txt' 


这 样 就 创建 出 一 个 空 的 文本 文件 10g3.txt， 以 后 输出 结果 不 仅 会 显示 在 屏幕 上 ， 还 会 写 人 log3. 
txt 中 。 男 外 ，log3.txt 会 保存 在 执行 命令 的 当前 路 径 中 。 拿 上 一 节 的 示例 来 说 ， 就 是 C 盘 的 
































Ci\Usersnisizawa 文件 夹 中 。 
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执行 tee 命令 后 ， 执 行 如 下 命令 。 


use db1 


SELECT * FROM tb; 
SELECT * FROM tbl; 








执行 结果 会 像 往常 一 样 显示 出 来 ， 内 容 会 写 人 log3.txt 中 。 


> 停止 向 文件 中 输出 执行 结果 
我 们 可 以 使 用 notee 命令 停止 向 文件 中 输出 执行 结 


notee 


mysql> use dbl 
Database changed 
mysql> SELECT * FROM tb; 

















+------- +------- +------- 十 
empid sales month 
+------- +------- +------- 十 
Al103 六 妇 二 4 
A1l02 54 S 
Al04 181 4 
Al01 184 4 
A103 17 S 
Al01 300 
A1l02 205 6 
Al04 93 Ss 
A1l03 12 6 
A1l07 87 6 
+------- +------- +------- 十 


10 rows in set (0.01 sec) 


mysql> SELECT * FROM tbl: 





























+------- +------ +------ + 
| empid | name | age | 
+------- +------ +------ + 
| Alol | 佐 芯 | 40 | 
| Al02 | 高 桥 | 28 | 
| aloa | 中川 | 20| 
| A104 | 渡 边 | 23 | 
| Alo5 | 泽 | 35 | 
+------- +------ +------ + 


5 rows in set (0.00 sec) 


mysql> notee 
Outfile disabled. 
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请 用 exit 等 命令 退出 MySQL 监视 器 。 使 用 文本 编辑 器 或 type 命令 确认 log3.txt 的 内 容 。 
MySQL 中 显示 的 内 容 变 为 文本 写 人 了 log3.txt 中 。 对 于 导出 的 文本 文件 ， 可 以 选择 需要 的 部 分 




















通过 “复制 ”一 “粘贴 ”等 菜单 操作 ， 在 其 他 系统 或 应 用 软件 上 使 用 。 





如 上 所 述 ，tee 和 notee 都 是 非常 方便 的 功能 。 


14.4 备份 和 恢复 数据 库 


14.4.1 备份 和 恢复 的 方法 





我 们 可 以 将 数据 库 的 设置 、 表 和 列 的 定义 、 数 据 等 数据 库 的 所 有 信息 作为 文件 导出 。 








> 转 储 


可 以 在 其 他 服务 器 上 创建 内 容 相同 的 数据 库 ， 也 可 以 备份 以 应 对 紧急 情况 的 发 生 。 








对 数据 库 的 所 有 内 容 执行 导出 的 操作 称 为 转 储 ( dump )( 见 图 14-6 )。 如 果 使 用 转 储 文件 ， 就 









































| ”恢复 
Be 





数据 库 的 全 部 内 容 


一 一 = MySsaL dump 10.13 Distrib 5.6.34, for Win64 (x86_64) 
-— 一 Host localhost Database: db1 
—— Server version 5.6.34-log 


/*!40101 SET @OLD_CHARACTER_SET_CLIENT=@@CHARACTER_SET_CLIENT */; 
/*!40101 SET @OLD_CHARACTER_SET_RESULTS=@@CHARACTER_SET_RESULTS */; 


—— Table structure for table "tb` 


DROP TABLE IF EXISTS "tb 
/*!40101 SET @saved_cs_client = @@character_set_client */; 
/*!40101 SET character_set_client = utf8 */; 
CREATE TABLE "tb ( 
"empid varchar(20) DEFAULT NULL, 
"sales int(11) DEFAULT NULL, 
"month int(11) DEFAULT NULL 
) ENGINE=InnoDB DEFAULT CHARSET=utf8; 
/*!40101 SET character_set_client = @saved_cs_client */; 


图 14-6 转 储 
我 们 可 以 在 命令 提示 符 上 执行 mysqldump 命令 来 转 储 MySQL 数据 库 。 
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使 用 文件 进行 交互 


mysqldump 命令 会 将 数据 库 的 配置 和 数据 本 身 作为 SQL 语句 写 出 来 ， 也 就 是 通过 “CREATE 


TABLE ...” 创 建 表 ， 然 后 写 出 “INSERT INTO ...” 这 样 的 SQL 语句 。 


通过 转 储 导 


信息 ， 可 以 说 “ 转 储 给 出 就 是 数据 库 本 映 ”。 


> 恢复 
把 通过 mys 


头 创建 数据 库 ， 其 实 就 是 将 包含 SQL 语句 集合 的 文本 文件 用 MySQL 命令 











出 的 信息 是 由 SQL 语句 生成 的 文本 。 通 过 这 些 文本 ， 我 们 可 以 读 取 数 据 库 的 所 有 














从 安全 方面 考虑 ， 我 们 需要 谨慎 对 待 这 些 信息 。 


qldump 命令 导出 的 数据 还 原 到 数据 库 中 的 操作 称 为 恢复 〈restore )。 恢 复 意 味 着 从 








进行 重 定向 。 


一 





试 着 使 用 mysqldump 命令 对 前 面 创 建 的 数据 库 dbl 的 所 有 信息 进 

















mysqlqdump 命令 的 执行 结果 写 入 文件 中 。 
转 储 数据 库 





























行 转 储 ， 并 通过 重 定向 将 











下 面 来 实际 操作 一 下 。 将 数据 库 dbl 的 信 ， 











区 执行 内 容 

































































息 转 储 到 名 为 db1_out.txt 的 文件 中 。 





/二 2 米 
> (执行 前 ) 数据 库 db1 
表 tb 
表 tb1 @ 一 一 一 一 数据 库 的 内 容 
视图 v1 
视 | V2 
存储 过 程 pr1 
存储 函数 fu1 
-- MySQL dump 10.13 Distrib 5.6.34, for Win64 (x86_64) 
-- Host: localhost Database: dbl 
-- Server version 5.6.34-1og 
/*140101 SET @OLD_CHRARACTER_SET_CLIENT=@Q@CHARRACTER_SET_CLIENT */; 
/*!40101 SET @OLD CHARACTER SET RESULTS=@@CHARACTER SET RESULTS */}; 
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/*!40101 SET @OLD COLLATION CONNECTION=@@COLLATION CONNECTION */; 

/*!40101 SET NAMES utf8 */; 

/*!40103 SET @OLD TIME ZONE=@@TIME ZONE */; 

/*!140103 SET TIME ZONE='+00:00' */; 

/*!40014 SET @OLD UNIQUE CHECKS=@@UNIQUE CHECKS, UNIQUE CHECKS=0 */; 

/*!40014 SET @OLD FOREIGN KEY CHECKS=@@FOREIGN KEY CHECKS, FOREIGN KEY CHECKS=0 */; 
/*!40101 SET @OLD SQL MODE=@@SQL MODE, SQL MODE='NO AUTO VALUE ON ZERO' */; 


/*!40111 SET @OLD SQL NOTES=@@SQL NOTES, SQL NOTES=0 */; 


-- Table structure for table “tb 


DROP TABLE IF EXISTS “tb ; 

/*!40101 SET @saved cs client = @@character set client */; 
/*!40101 SET character set client = utf8 */; 

CREATE TABLE ‘tb ( 

“empid™ varchar(20) DEFAULT NULL, 

“sales™ int(11) DEFAULT NULL, 

“month™ int(11) DEFAULT NULL 

) ENGINE=InnoDB DEFAULT CHARSET=utf8; 


/*!40101 SET character set client = @saved cs client */; 


-- Dump completed on 2018-08-04 15:54:51 


】 


转 储 ( 文本 文件 db1_out.txt ) 























操作 方法 











在 命令 提示 符 上 执行 下 面 的 命令 。 




















mysqldump -u root -proot dbl>dbl1 out.txt 


C:\Users\keglu>mysqldump -u root -proot dbl>dbl1 out.txt 























需要 花 一 点 时 间 才 能 完成 。 由 于 输出 的 是 一 个 文本 文件 ， 所 以 我 们 可 以 使 用 编辑 器 确认 一 下 它 
的 内 容 。 另 外 ， 因 为 mysqldump 的 执行 结果 是 用 MySQL 服务 器 的 默认 字符 编码 输出 的 ， 所 以 在 
本 书 的 环境 中 字符 编码 为 UTF-8。 因 此 ， 如 果 使 用 type 命令 确认 db1_out.txt 的 内 容 ， 中 文部 分 会 
乱码 ， 这 一 点 需要 注意 。 
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db1_out.txt 的 部 分 内 容 





-- MySQL dump 10.13 Distrib 5.6.34, for Win64 (x86 64) 


-- Host: localhost Database: dbl 





/*!140101 SET @OLD CHARACTER SET CLIENT=@@CHARACTER SET CLIENT */; 
/*!140101 SET @OLD CHARACTER SET RESULTS=@@CHARACTER SET RESULTS */; 
/*!140101 SET @OLD COLLATION CONNECTION=@@COLLATION CONNECTION */; 
/*!140101 SET NAMES utf8 */; 
/*!140103 SET @OLD TIME ZONE=@@TIME ZONE */; 
/*!140103 SET TIME ZONE='+00:00' */; 
/*!140014 SET @OLD UNIQUE CHECKS=@@UNIQUE CHECKS, UNIQUE CHECKS=0 */; 

/*!140014 SET @OLD FOREIGN KEY CHECKS=@@FOREIGN KEY CHECKS, FOREIGN KEY CHECKS=0 */; 
/*!40101 SET @OLD SQL MODE=@@SQL MODE, SQL MODE='NO AUTO VALUE ON ZERO' */; 
/*!140111 SET @OLD SQL NOTES=@@SQL NOTES, SQL NOTES=0 */; 
























































-- Table structure for table ‘tb 


DROP TABLE IF EXISTS ‘tb; 

/*!40101 SET @saved cs client = @@character set client */; 
/*140101 SET character set client = utf8 */; 

CREATE TABLE “tb ( 

“empid™ varchar(20) DEFAULT NULL, 

~sales™ int(11) DEFAU NULL, 

month™ int(11) DEFAULT NULL 
) ENGINE=InnoDB DEFAULT CHARSET=utf8; 

/*!40101 SET character set client = @saved cs client */; 






































-- Dump completed on 2018-08-04 15:54:51 











行 的 开头 是 “- -” 的 部 分 是 注释 ， 用 “/*” 和 “*/” 括 起 来 的 部 分 也 是 注释 (一 15.8.3 节 ),， 注 
释 不 是 命令 而 是 单纯 的 “说 明 ”。 上 面 的 执行 结果 中 有 命令 “CREATE TABLE ...”， 它 用 于 创建 表 。 




















恢复 作业 失败 的 情况 


如 果 在 之 后 的 操作 中 恢复 作业 失败 了 ， 试 着 在 转 储 时 加 上 --default-character-set=utf8 之 类 的 
字符 编码 的 选项 , 执行 mysqldump -u root -proot dbl > dbl out.txt --default-character- 
set=utf8。 具 体内 容 请 参考 3.4.3 节 。 






































14.4 备份 和 恢复 数据 库 


和 一 


接 下 来 恢复 转 储 文件 。 在 命令 提示 符 上 使 用 重 定 向 将 文件 还 原 到 数据 库 。 


这 次 是 在 同一 台 计 算 机 上 对 数据 库 
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进行 复制 ， 当 然 我 们 也 可 以 将 数据 库 恢复 到 其 他 的 MySQL 


环境 。 当 恢复 数据 库 时 ， 我 们 必须 提前 准备 一 个 用 于 填充 的 数据 库 。 如 果 没 有 数据 库 ， 就 需要 提前 


创建 一 个 "。 


下 面 来 恢复 转 储 文件 。 创 建 数据 库 db2， 然 后 将 转 储 文件 db1_out.txt 恢复 到 该 数据 库 中。 


PP 执行 内 容 


> 创建 数据 库 db2 
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> 文本 文件 db1_out.txt 











将 该 文本 文件 恢复 到 数据 库 db2 中 

















/*!140101 
/*!40101 
/*!140101 
/*!140103 
/*140103 
/*!140014 
/*!40014 
CHECKS=0 
/*!40101 
x 

/*!140111 





/*!40101 SE 
SE 
SE 
SET 
SE 
SET 
SET 
SE 


SE 


SET 


| 





| 





中 在 没有 数据 库 的 情况 下 会 报错 。 


| 


-- MySQL dump 10.13 Distrib 5.6.34, for Win64 (x86 64) 


: localhost Database: dbl 


-- Server version 5.6.34-1log 














@OLD CHARACTER SET CLIENT=@@CHARACTER SET CLIENT */; 
@OLD CHARACTER SET RESULTS=@@CHARACTER SET RESULTS */; 
@OLD COLLATION CONNECTION=@@COLLATION CONNECTION */; 
NAMES utf8 */; 


@O] 


| 





D_TIME ZONE=@@TIME ZONE */; 


TIME ZONE='+00:00' */; 


@O] 
@O] 


@O 


@O 


| 


| 


D UNIQUE CHECKS=@@UNIQUE CHECKS, UNIQUE CHECKS=0 */; 


D_FOREIGN KEY CHECKS=@@FOREIGN KEY CHECKS, FOREIGN KEY_ 


ID SQL MODE=@@SQL MODE, SQL MODE='NO AUTO VALUE ON ZERO' 




















D_SQL NOTES=@@SQL NOTES, SQL NOTES=0 */; 





译 者 注 
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-- Table structure for table “tb 


DROP TABLE IF EXISTS “tb ; 
/*!40101 SET @saved cs client = @@character set client */; 





/*!40101 SET character set client = utf8 */; 
CREATE TABLE “tb ~ ( 

“empid varchar(20) DEFAULT NULL, 

~sales” int(11) DEFAULT NULL, 
~month” int(11) DEFAULT NULL 
) ENGINE=InnoDB DEFAULT CHARSET=uUutf8; 

/*!40101 SET character set client = @saved cs client */; 
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> (执行 后 ) 创建 数据 库 db2 


表 tb 
表 tb1 
视图 v1 


























视图 v2 
存储 过 程 pr1 
存储 函数 fu1 





























操作 方法 


中 在 命令 提示 符 上 执行 下 面 的 命令 。 于 是 创建 出 数据 库 db2 ( 一 7.9 节 )。 


























一 般 情况 下 ， 按 照 上 述 方法 操作 就 能 成 功 复制 数据 库 db2。 发 生 错 误 的 话 ， 可 以 在 字符 编码 上 
找 找 原因 。 这 时 可 以 按照 接 下 来 介绍 的 方法 指定 字符 编码 进行 转 储 和 恢复 。 
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Ce 14.4.4 字符 编码 问题 


在 大 部 分 情况 下 ， 以 上 操作 能 顺利 执行 。 不 过 ,在 使 用 MySQL 的 过 程 中 ， 我 们 经 常会 受到 字 
符 编码 问题 的 困扰 。 

如 果 单 纯 进 行 “ 转 储 一 恢复 ”， 像 汉字 这 种 需要 占用 两 个 字 节 的 字符 可 能 就 会 出 现 问 题 。 这 是 
因为 在 进行 转 储 和 恢复 的 时 候 ， 如 果 不 指 定 dd dt 

如 果 发 生 错误 从 而 无 法 顺利 进行 “ 转 储 一 恢复 ”， 这 时 就 可 以 尝试 指定 字符 编码 进行 转 储 和 1 

指定 字符 编码 的 选项 如 下 所 示 。 


--default-character-set= 字 符 编 码 


例如 ， 在 指定 表示 UTF-8 的 utf8 的 情况 下 ， 要 加 上 --default-character-set=utf8 选项 。 
下 面 的 示例 使 用 了 该 选项 将 字符 编码 指定 为 UTF-8 进行 转 储 和 恢复 。 将 数据 库 转 储 到 db3_out. 
xt， 并 恢复 到 数据 库 db3 中 。 









































Dy 





mysqldump -u root -proot dbl>db3 out.txt --default-character-set=utf8 


mysql -u root -proot db3<db3 out.txt --default-character-set=utf8 





锁 表 
下 面 将 介绍 通过 给 表 加 锁 ( 一 13.4.3 节 ) 来 限制 表 的 操作 的 LOCK 命令 。 


LOCK 命令 通过 给 指定 的 表 加 上 某 种 类 型 的 锁 来 限制 对 表 的 操作 。 锁 的 类 型 主要 有 以 下 几 种 ( 表 14-1 )。 
表 14-1 锁 的 类 型 

锁 的 类 型 限制 内 容 

READ 所 有 客户 端 都 只 允许 执行 SELECT。 只 读 锁 


READ LOCAL ( ※ 如 果 是 本 书 使 用 | 对 于 InnoDB 以 外 的 存储 引擎 ， 加 锁 的 客户 端 仅 能 执行 SELECT 。 
的 InnoDB， 则 和 上 面 的 READ 相同 ) | 本 地 只 读 锁 


WRITE 没有 加 锁 的 客户 端 不 能 进行 任何 操作 ， 拥 有 锁 的 客户 端 可 以 执行 操作 













































































给 表 my_table 设置 READ 锁 时 需要 执行 如 下 命令 。 





Q 对 于 InnoDB 以 外 的 存储 引擎 ，READ LOCAL 允许 加 锁 以 外 的 客户 端 执行 非 冲突 性 INSERT 语句 ， 但 是 不 允许 执 


译 者 注 
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LOCK TABLES my table READ; 


另外 ，UNLOCK 命令 用 于 给 表 解 锁 。 




















给 表 解 锁 


我 们 可 以 执行 下 面 的 命令 解锁 当前 所 有 的 表 。 





UNLOCK TABLES; 


本 章 介 绍 了 以 下 内 容 。 
@ 使 用 文本 文件 导入 和 导出 数据 的 方法 
@ 执 行 包 含 SQL 语句 的 文件 的 方法 


@ 将 查询 的 结果 保存 到 文件 中 的 方法 
@ 备 份 和 恢复 整个 数据 库 的 方法 


在 实际 工作 中 ， 我 们 很 少 使 用 INSERT 命令 逐个 输入 大 量 数据 。 大 家 一 定 要 记 住 使 用 文件 处 理 
数据 的 方法 。 





> 自我 检查 
下 面 检查 一 下 本 章 学 习 的 内 容 是 否 全 部 理解 并 掌握 了 。 




















口 能 够 通过 “LOAD DATA INFILE ...” 导 入 数据 

口 能 够 通过 “sELECT * INTO OUTFILE ...” 导 出 数据 
口 能 够 使 用 SOURCE 命令 批量 执行 SQL 语句 

口 能 够 使 用 重 定向 导出 执行 结果 

口 能 够 使 用 tee 命令 导出 执行 结果 

口 能 够 使 用 mysqldump 命令 备份 整个 数据 库 

口 能够 使 用 重 定向 恢复 数据 库 。 





国 练 习题 








问题 1 
mysqlqdump 命令 可 以 以 特定 的 表 为 对 象 进 行 备 份 和 恢复 。 当 备份 特定 的 表 时 ， 需 要 使 用 命令 提示 符 按 
照 以 下 格式 执行 命令 。 





























国 表 的 备份 




















另外 ， 与 恢复 整个 数据 库 时 一 样 ， 我 们 可 以 用 MySQL 命令 对 备份 的 文本 文件 执行 重 定 向 。 
请 使 用 以 上 方法 备份 数据 库 db1 中 的 表 tb1， 然 后 删除 表 tb1， 再 将 其 还 原 。 用 户 名 是 root， 密 码 是 
root， 备 份 之 后 创建 的 文本 文件 的 名 称 是 test.txt。 

























































































ta 参考 答案 


问题 1 














J 在 命令 提示 符 上 执行 下 面 的 命令 ， 备 份 表 tb1。 


mysqldump -u root -proot dbl tbl1 > test.txt 


C:\Users\nisizawa>mysqldump -u root -proot dbl tbl > test.txt 














@ 启动 MySQL 监视 器 选择 数据 库 dbl 后 ， 执 行 下 面 的 命令 ， 删 除 表 tb1。 


DROP TABLE tbl; 


mysql> DROP TABLE tbl; 
Query OK, 0 rows affected (0.09 sec) 














(3) 在 命令 提示 符 上 执行 下 面 的 命令 ,恢复 表 tb1。 


mysql -u root -proot dbl < test.txt 
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C:\Users\nisizawa>mysql -u root -proot dbl < test.txt 

















下 面 确 认 一 下 表 是 否 存 在 。 
启动 MySQL 监视 器 选择 数据 库 dbl 后 ， 执 行 如 下 命令 ， 确 认 表 tb1。 


SELECT * FROM tb1; 
























































执行 结果 
mysql> SELECT * FROM tbl: 
Fs 二 生 == 一 = 一 = + 
empid name age 
+ Eee Ee + 
A1l01 佐藤 40 
A102 高 28 
A103 中 川 20 
RA104 渡 边 23 
Al1L05 泽 35 
HE 和 让 + 


5 rows in set (0.00 sec) 











第 4 部 分 
MySQL+PHP 的 基础 


用 于 控制 MySQL 的 PHP 


PHP 基 础 知识 


PHP 脚 本 和 HTML 


使 用 PHP 脚 本 操作 MySQL 











本 书 是 一 本 从 基础 开始 学 习 MySQL 的 书 。 但 现在 要 想 真 正 掌握 MySOL， 还 需要 将 Web 
纳入 学 习 的 范围 内 。 本 书 将 介绍 使 用 MySQL 必须 掌握 的 Web 和 PHP 的 知识 。 
在 第 4 部 分 中 ， 我 们 将 学 习 PHP 的 基础 知识 。 
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在 本 章 之 前 ， 我 们 一 直 在 CUI 环境 的 MySQL 监视 器 上 执行 
MySOL。 相 信 不 少 人 觉得 数据 库 有 些 枯燥 乏味 。 

MySOL 的 世界 不 只 有 MySQL 监视 器 。 从 本 章 开 始 ， 我 们 
将 使 用 平常 都 能 接触 到 的 Web 浏览 器 来 操作 MySQL。 让 我 们 
更 用 功能 强大 的 最 佳 组 合 “Apache + MySOL + PHP” 来 学 习 
MySQL 的 各 种 使 用 方法 吧 | 













































































15.1 要 创建 的 示例 


在 本 章 和 第 16 章 中 ， 我 们 将 学 习 用 于 控制 MySQL 的 脚本 语言 PHP 以 及 HTML 的 基础 知识 。 
如 果 使 用 本 章 出 现 的 脚本 ， 就 可 以 创建 出 如 图 15-1 所 示 的 Web 页 面 ( 这 里 还 没有 使 用 MySQL )。 

















sm x 


€) © localhost/dream.php 





下 载 示 例 中 有 一 个 名 为 to_htdocs 
的 文件 夹 ， 这 是 保存 在 该 文件 夹 
中 的 boy.gif 图 像 。 可 以 按照 自己 
的 喜好 放置 该 图 像 













































































根据 时 间 变 化 的 消息 
(一 16.6.3 节 ) 


























当前 日 期 ( 一 16.3.2 节 ) 

















显示 客户 端的 IP 地 址 、 主 机 名 
和 浏览 器 名 称 ( 一 16.3.3 节 ) 

















POST 发 送 (一 17.9.1 节 ) 














15-1 最 终 版 页 面 dream.php 
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在 创建 示例 的 时 候 ， 想 好 自己 将 来 要 发 布什 么 样 的 网 页 ， 在 此 基础 上 自由 地 进行 设计 。 





15.2 在 Web 应 用 程序 中 使 用 MySQL 


前 面 我 们 学 习 了 使 用 MySQL 的 方法 。 通 过 在 MySQL 监视 器 上 用 键盘 输入 SELECT 和 
INSERT 等 SQL 命令 ， 初 次 接触 了 MySQL。 也 就 是 说 ， 如 果 想 使 用 MySQL 数据库， 就 得 先 学 会 
使 用 SQL。 

如 果 操 作 MySQL 的 方法 只 此 一 和 种， 恐怕 MySQL 不 会 在 世界 范围 内 被 这 么 广泛 地 使 用 了 。 

除了 那些 枯燥 无 味 的 使 用 方法 ， 我 们 还 可 以 创建 应 用 程序 ， 点 击 浏览 器 上 的 按钮 来 操作 
MySQL 数据 库 。 也 就 是 说 ， 点 击 按钮 就 会 执行 实现 某 种 功能 的 SQL 语句 。 如 果 创 建 了 这 样 一 个 机 
制 ， 那 么 使 用 者 即使 不 懂 复 杂 的 SQL 语句 ， 也 能 在 浏览 嚣 中 使 用 数据 库 。 实 际 上 ， 论坛 、 网 店 、 
销售 管理 系统 以 及 学 校 的 教务 管理 系统 等 多 个 领域 都 使 用 了 MySQL。 

像 这 样 ， 将 浏览 器 当 作用 户 界 面 并 在 Web 服务 器 端 进行 配置 ， 通 过 网 络 来 操作 MySQL 等 的 系 
统称 为 Web 应 用 程序 。 创 建 Web 应 用 程序 需要 用 到 编程 语言 ， 而 MySQL 支持 Perl、C、PHP 和 
Java 等 非常 多 的 编程 语言 。 














































































































15.3 ”使 用 Web 时 需要 用 到 的 机 制 


全 15.3.1 Web 服务 器 和 客户 端 
我 们 来 试 着 思考 一 下 使 用 Web 时 需要 用 到 什么 样 的 机 制 。 点 击 般 入 在 Web 页 面 上 的 超 链接 ， 
可 以 获得 全 世界 的 信息 。 超 链接 中 有 “x x 地 方 的 x x 文件 ”等 信息 。 

















P Web 服务 器 


这 个 “x x 地 方 ”表示 特定 的 Web 服务 器 。Web 服务 器 是 连接 到 互联 网 的 机 器 ， 设 置 了 用 于 
实现 服务 器 功能 的 应 用 程序 等 内 容 。 服 务 器 在 被 访问 时 ， 会 按照 要 求 返回 保存 的 数据 ， 或 者 执行 指 
示 的 处 理 并 返回 结 

















P” 客户 端 
使 用 Web 服务 器 的 用 户 的 计算 机 称 为 客户 端 。 当 用 户 点 击 超 链接 时 ， 客 户 端 将 向 指定 的 Web 
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服务 器 发 送 “ 发 送 了 x x !” 这 样 的 信息 。Web 服务 器 收 到 请 求 后 ， 会 把 指定 的 文本 和 图 像 等 数据 
发 送 给 客户 端 ， 也 就 是 发 送 到 大 家 使 用 的 Web 浏览 器 上 ( 见 图 15-2 )。 














15-2 ”Web 服务 器 和 客户 端 


15.3.2 ”Web 服务 器 的 作用 


Web 服务 器 和 客户 端 通过 HTTP (HyperText Transfer Protocol， 超 文本 传输 协议 ) 进行 通信 。 
协议 是 计算 机 之 间 进 行 通信 时 需要 共同 遵循 的 规则 。 

Web 网 页 的 URL 写 为 “http:/…”， 这 个 http:// 的 部 分 用 于 声明 使 用 了 HTTP 协议 进行 通信 。 
Web 服务 器 有 “如 果 客 户 端 发 送 了 基于 HTTP 协议 的 请 求 ， 则 将 相应 的 文件 和 图 像 发 送 过 去 ”的 功 
能 。 像 HTTP 这 种 在 “请 求 发 送 一 已 发 送 一 结束 ”后 立即 终止 通信 的 协议 称 为 无 状态 协议 〈 stateless 


protocol )。 








G15.3.3 Apache 和 Web 服 务 部 


Web 服务 顺 的 功能 由 作为 服务 器 连接 到 互联 网 的 计算 机 上 的 软件 负责 处 理 。Apache 软件 基金 
会 的 Web 服务 器 软件 Apache(〈 阿 帕 奇 ) 非常 有 名 ,广泛 使 用 在 全 世界 的 很 多 Web 服务 器 上 。 
Apache 和 MySQL 一 样 是 开源 的 ， 可 以 免费 使 用 。 世 界 各 地 的 志愿 者 每 天 都 在 改善 Apache 的 代码 。 
用 于 修改 和 优化 代码 的 程序 称 为 补丁 (patch )。 据 说 ，Apache 是 通过 收集 补丁 开发 出 来 的 Web 服务 
需 软 件 ， 所 以 叫 作 Apache。 




















© The Apache HTTP Server Project 
http://httpd.apache.org/ 





Apache 是 与 MySQL 以 及 后 面 介绍 的 PHP 兼容 性 最 好 的 Web 服务 器 软件 。 本 书 的 导入 示例 中 
使 用 的 是 MAMP， 在 这 种 情况 下 Apache 就 需要 作为 Web 服务 器 软件 使 用 。 
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有 状态 协议 
HTTP 是 无 状态 协议 ， 而 在 Web 上 进行 文件 传输 时 使 用 的 FTP ( File Transfer Protocol， 文 件 传输 协议 ) 等 
需要 维持 连接 状态 的 协议 称 为 有 状态 协议 。 









































15.4 静态 页 面 和 动态 页 


最 初 的 Web 页 面 ， 只 具备 点 击 超 链 接 后 相应 的 文件 就 能 发 送 过 来 的 功能 ( 见 图 15-3 )。 这 种 机 
制 称 为 静态 页 面 。Web 页 面 是 按照 HTML 格式 编写 的 文本 文件 (一 17.1 节 )。 

而 最 近 动 态 的 Web 页 面 变 得 越 来 越 多 。 在 动态 Web 页 面 中 ， 服 务 器 可 以 处 理 客户 端 发 送 的 数 
据 ， 并 将 相应 的 Web 页 面 显 示 到 客户 端 ( 见 图 15-3 )。 

通常 我 们 会 使 用 Perl、Java 和 PHP 等 编程 语言 来 实现 这 种 “服务 器 端的 处 理 ” 功 能 。 






















































































客户 端 Web 服 务 器 














http://O O.htm 
































A 
客户 端 PHP 脚 本 人 


- 处 理 后 


<html> 
<head><title> 
末 窒 以 求 的 …… 
shooy 




















Ls 





http://O O.htm 






图 15-3 静态 Web 页 面 和 动态 Web 页 面 
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15.5 ”在 Web 上 运行 的 程序 


关于 在 Web 上 运行 的 程序 的 机 制 ，CGI (Common Gateway Interface， 公 共 网 关 接口 ) 和 脚本 
比较 有 名 。 


C 15.5.1 CGI 
CGI 是 将 程序 放置 在 服务 器 上 ， 通 过 响应 来 自 Web 浏览 器 的 调用 来 执行 程序 的 机 制 。 能 够 











用 来 创建 CGI 程序 的 语言 有 很 多 ， > Perl 比较 有 名 。Perl 在 很 早 以 前 就 用 于 创建 访问 计数 器 等 应 
用 了 。 








15.5.2 脚本 

脚本 最 初 指 的 是 为 了 自动 执行 处 理 而 创建 的 简单 程序 。 脚 本 以 各 种 各 样 的 形式 存在 ， 比 如 单独 
创建 成 一 个 文本 文件 ， 或 者 编写 在 HTML 文件 中 等 。 

在 Web 上 运行 的 脚本 ,通常 是 指 将 脚本 的 内 容 包含 在 HTML 文件 中 ， 并 根据 需要 使 之 运行 的 
程序 。 

目前 ， 在 Web 上 使 用 的 常见 脚本 有 “在 客户 端 运行 的 脚本 ”和 “在 Web 服务 器 端 运行 的 脚本 ” 
两 种 ( 见 图 15-4 )。 











P 客户 端 脚本 

客户 端 脚本 是 指 JavaScript 等 在 客户 端 上 运行 的 脚本 。 与 Web 服务 器 完全 无 关 ， 程 序 会 在 浏览 
网 页 的 个 人 计算 机 上 执行 。 也 就 是 说 ， 对 于 客户 端 脚本 ， 如 果 Web 服务 器 将 脚本 发 送 到 客户 端 ， 
之 后 就 不 能 做 其 他 任何 事情 了 。 

因为 客户 端 脚本 是 在 客户 端的 环境 上 运行 的 ， 所 以 很 容易 控制 浏览 器 上 的 显示 和 操作 。 但 是 ， 
不 同 种 类 的 浏览 器 运行 效果 也 会 出 现 差异 ， 有 时 会 出 现 无 法 运行 的 情况 。 









































BP 服务 器 端 脚本 

服务 器 端 脚 本 在 Web 服务 器 上 执行 。 当 接收 到 来 自 客户 端的 命令 后 ，Web 服务 器 上 会 完成 处 
理 ， 并 将 处 理 结果 发 送 给 客户 端 。 客 户 端 只 是 用 于 查看 处 理 结果 而 已 。 数 据 库 的 处 理 通常 会 在 服务 
器 端 进行 ， 因 此 很 适合 使 用 服务 器 端 脚本 。 
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< 客户 端 脚本 > 


客户 端 服务 器 端 


二 4 VW 


function idou() 
if(x>200 || x<O){ 
document.d1.style.left 
document.d1.style.top 





SD 


《 服务 器 端 脚本 > 


oe , 
~ $s=new PDO("'mysql:host= 


localhost;dbname=db1", 
"root","root"); 
$q=<<<eot 














SELECT bang,AVG(uria) 
FROM tb 
WHERE uria>=50 





15-4 客户 端 脚本 和 服务 器 端 脚本 


15.6 什么 是 PHP 


本 书 将 介绍 PHP 作为 创建 Web 应 用 程序 的 编程 语言 使 用 的 示例 。 





人 15.6.1 PHP 是 什么 


PHP 是 在 Web 服务 器 端 运行 的 服务 器 端 脚本 。 因 为 执行 的 模块 〈 最 小 单元 程序 ) 在 Web 服务 
器 上 且 通 过 脚本 的 命令 运行 ， 所 以 一 般 来 说 运行 速度 很 快 。 








个 PHP : Hypertext Preprocessor 
http://php.net/ 


PHP 的 全 称 是 Hypertext Preprocessor， 一 种 用 于 开发 Web 应 用 程序 的 编程 语言 ， 支 持 
Apache 和 微软 公司 开发 的 IIS ( Internet Information Services， 互 联网 信息 服务 ) 等 许多 Web 服务 
器 ， 并 支持 包括 MySQL 在 内 的 许多 RDBMS。 最 重要 的 是 ， 它 是 一 门 非常 简单 的 语言 ， 使 用 该 语 
言 可 以 轻松 创建 脚本 。 

PHP 应 用 于 世界 各 地 的 Web 服务 器 中 。 可 以 说 PHP 是 创建 Web 应 用 程序 最 常用 的 编程 语言 之 
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S 15.6.2 本 书 使 用 的 PHP 
本 书 介绍 的 PHP 知识 是 在 Web 应 用 程序 中 处 理 MySQL 所 需 的 最 低 限度 的 基础 知识 。PHP 的 
功能 非常 多 ， 本 书 无 法 一 一 介绍 。 这 里 只 选取 如 下 内 容 进行 讲解 。 





3 如 何 使 用 PHP 发 布 前 14 章 学 习 的 SQL 命令 




















大 家 可 以 根据 本 书 的 内 容 来 试 着 构建 以 大 规模 运用 为 前 提 的 数据 库 服务 顺 ， 也 可 以 试 着 使 用 租 
借 服 务 器 。 但是， 这 和 理解 MySQL 的 本 质 以 及 进行 实际 操作 并 没有 太 大 的 关系 。 

后 面 我 们 将 使 用 公告 板 作 为 示例 。 示 例 虽 然 简单 ， 但 是 通过 这 个 例子 我 们 也 可 以 学 到 MySQL 
使 用 方面 的 精髓 ， 有 利于 为 今后 的 学 习 打下 良好 的 基础 。 请 一 步 一 步 进行 操作 ， 一 边 实 践 一 边 阅 读 。 





























S 15.6.3 ”设置 php.ini 

在 学 习 PHP 之 前 ， 我 们 需要 对 php.ini 配置 文件 设置 用 于 处 理 日 期 、 时 间 的 时 区 ， 以 及 多 字 节 
字符 串 的 相关 内 容 。 

php.ini 是 用 于 设置 PHP 动作 的 文本 文件 ， 在 本 书 使 用 的 环境 中 ， 它 位 于 文件 夹 C:MAMP\conf\ 
php7.1.5 中 。 在 conf 文件 夹 中 ，PHP 的 每 一 个 版 本 都 分 成 了 一 个 文件 来 ， 注 意 不 要 弄 错 版 本 。 可 以 
点 击 MAMP 起 始 页 面 中 的 “phpinfo” 来 确认 PHP 的 版 本 (一 2.2.4 节 )。 






































P 设置 时 区 

在 MAMP 的 PHP 中 ,时 区 的 默认 值 是 UTC。UTC 是 决定 世界 各 地 时 间 标 准 的 “世界 标准 时 
间 ”， 和 北京 时 间 约 差 8 小 时 。 因 此 ， 如 果 使 用 date 函数 (一 16.3.2 节 ) 显示 时 间 就 会 出 现 偏差 ， 
所 以 需要 重 置 时 区 。 

使 用 文本 编辑 器 打开 php.ini 文件 ， 删 除 第 703 行 附近 的 “; date .timezone =” 开 头 的 
“;”， 并 将 值 设置 为 Asia/shanghai"。 






































[Date] 
; Defines the default timezone used by the date functions 
date.timezone = Asia/Shanghai 人 州 除 ';”, 添加 "Asia/shanghai” 


P 设置 多 字 节 字符 串 
在 本 书 使 用 的 PHP 7.1.5 中 ， 字 符 编 码 上 默认 设置 为 UTF-8， 所 以 我 们 不 需要 对 字符 编码 进行 设 
置 。 动 态 生 成 的 字符 串 将 自动 以 UTF-8 进行 输出 。 





























四 注意 ，date.timezone 这 个 参数 从 PHP 5.1.0 开始 有 效 。 对 中 国 时 区 来 说 ， 能 够 设置 的 值 有 Asia/Shanghai、 
译 者 注 





Asia/Taipei 和 Asia/Hong Kong 等 , 没有 Asia/Beijing 这 个 值 。 
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但 是 ， 当 处 理 中 文 时 ， 如 果 不 进行 多 字 节 字符 串 ( 汉字 等 两 个 字 节 以 上 的 字符 串 ) 的 相关 设 
置 ， 在 函数 的 返回 值 包含 中 文 的 情况 下 就 可 能 会 发 生 乱 码 。 
所 以 ， 请 删除 php.ini 文件 第 1232 行 附近 的 “;mbstring.language = Japanese” 开 头 的 
;”， 并 将 Japanese 修改 为 Chinese。 








[mbstring] 
; language for internal character representation. 


mbstring.language = Chinese 删除 开头 的 ";”"， 并 修改 为 Chinese。 





完成 上 述 修改 后 保存 php.ini， 重 新 启动 服务 右 (一 2.2.3 节 )。 
我 们 也 可 以 在 php.ini 中 设置 其 他 内 容 。 这 部 分 知识 就 不 进行 讲解 了 ， 感 兴趣 的 读者 请 参考 
PHP 的 专业 用 书 。 

















15.7 首先 显示 “欢迎 光临 !” 


15.7.1 确认 Apache 是 否 启动 


Apache 必须 处 于 启动 状态 。 启 动 Apache 的 方法 会 根据 使 用 的 环境 发 生变 化 。 如 果 安 装 了 本 书 
第 一 章 中 介绍 的 MAMP， 就 可 以 参考 2.2.3 节 中 介绍 的 方法 来 操作 。 

下 面 确认 一 下 Apache 是 否 处 于 启动 状态 。 在 浏览 器 的 地 址 栏 中 输入 http://localhost/MAMP， 然 
后 按 Enter 键 。 
如 果 按 照 本 书 2.2.2 节 介 绍 的 步骤 进行 了 安装 和 设置 ， 访 问 http://localhost/MAMP 应 该 就 能 显 
示 出 图 15-5 的 画面 了 。 







































































| 园 napvnocalhosyMAMP/ Pr-dc| 园 MAMp 


Buy MAMP PRO 


和 Welcome! 人 
MAMP has been 
installed 


successfully. 





图 15-5 MAMP 的 开始 页 
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接 下 来 要 创建 的 PHP 文件 也 可 以 在 网 上 公开 。 不 过 我 们 先 试 着 在 相当 于 Web 服务 器 的 计算 机 
上 使 用 它 。 














localhost 的 IP 地址 
localhost 的 IP 地 址 通常 被 指定 为 “127.0.0.1"。 也 就 是 说 ， 在 常规 设置 中 ， 输 入 http://127.0.0.1/MAMP， 
就 会 显示 出 图 15-5 的 画面 。 









































S 15.7.2 ”首先 用 PHP 显示 “欢迎 光临 !” 
确认 Apache 可 以 顺利 工作 之 后 ， 试 着 用 PHP 显示 “欢迎 光临 1”"。 操 作 过 程 中 可 能 会 遇 到 显示 
不 出 来 或 乱码 等 情况 ， 我 们 来 一 个 一 个 地 解决 这 些 问 题 。 


区 使 用 哪个 文本 编辑 器 


请 使 用 支持 UTF-8 的 编辑 器 创建 PHP 脚本 。 在 本 书 中 ，PHP 脚本 和 HTML 的 字符 编码 使 用 了 
UTF-8。 
虽然 使 用 Windows 的 记事 本 也 可 以 读 写 UTF-8 的 文本 文件 ， 但 是 在 使 用 记事 本 的 情况 下 ， 
UTF-8 文件 会 附加 BOM。BOM 是 文件 开头 附加 的 显示 Unicode (UTF-8 等 ) 种 类 的 标志 。 在 附 有 
BOM 的 情况 下 ， 某 些 程序 可 能 会 发 生意 外 错误 ， 因 此 笔者 不 建议 使 用 记事 本 。 
由 GitHub 开发 的 Atom 将 UTF-8 (无 BOM ) 作为 默认 的 字符 编码 ， 当 编写 PHP 程序 时 可 以 使 
代码 辅助 等 功能 ， 非 常 方便 。 


个 Atom 
https://atom.io/ 


Atom 是 开源 的 ， 可 以 免费 使 用 。 另 外 ，Atom 上 有 各 种 各 样 的 附加 程序 以 包 (package ) 的 形 
式 提 供 ， 因 此 可 以 自 定 义 首选 项 。Atom 的 画面 如 图 15-6 所 示 。 
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(ND dientphp — CNMAME aoe A 一 


File Edit Vi n Find Packages Help 





在 这 里 选择 字符 编码 














在 这 里 选择 PHP 就 可 以 
使 用 代码 辅助 功能 



































图 15-6 ”Atom 的 画面 
下 面试 着 创建 一 个 PHP 脚本 。 请 启动 能 够 兼容 UTF-8 的 编辑 器 。 局 动 后 ， 输 入 代码 清单 15-1 
的 内 容 。 注 意 ， 只 有 “欢迎 光临 !” 的 部 分 是 中 文 ， 其 他 内 容 务必 使 用 半角 输入 。 


代码 清单 15-1 test.php 



























































<?php 
Print " 欢迎 光临 ! "; 


二 














我 们 将 在 第 16 章 学 习 编写 PHP 脚本 的 语法 规则 。 现 在 先 想 办 法 确保 PHP 脚本 能 够 有 效 运行 。 

将 创建 好 的 文件 保存 在 Web 服务 絮 发 布 的 文件 夹 中 (一 2.2.5 市 )。 存储 位 置 会 根据 Apache 的 
设置 发 生变 化 ， 在 本 书 的 环境 中 为 C\MAMP\htdocs 文件 夹 (在 Windows 的 情况 下 ),。 今后 要 创建 
的 PHP 脚本 文件 也 会 保存 在 这 里 。 








| htdocs 0 在 这 个 文件 夹 下 创建 文件 
一 一 test.php 


保存 好 后 就 可 以 执行 PHP 脚本 了 。 在 浏览 器 的 地 址 栏 中 输入 如 下 内 容 ，testphp 就 可 以 运行 了 。 


http://localhost/test.php 


像 图 15-7 这 样 显示 出 “欢迎 光临 !” 的 字样 就 表示 运行 成 功 。 
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回 http://localhost/testphp 内 ~ © | 国 localhost 次 价 次 © 


欢迎 光临 ! 











图 15-7 运行 test.php 的 示例 
如 果 没 有 正常 显示 ， 就 需要 参考 下 一 节 的 内 容 来 解决 问题 。 如 果 能 够 正常 显示 出 来 ， 就 可 以 开 
台 学 习 15.8 节 的 内 容 了 。 

另外 ， 如 果 机 器 的 JP 地址 是 “192.168.1.1”， 在 网 络 内 的 其 他 客户 端的 浏览 器 中 输入 
http://192.168.1.1/test.php， 也 会 显示 图 15-7 的 画面 。 但 是 ， 受 到 环境 安全 状况 等 因素 的 影响 ， 也 可 
能 无 法 从 除 localhost 以 外 的 地 方 进行 访问 。 

此 外 ， 文件 名 设置 成 了 以 “.php” 为 扩展 名 的 test.php。 这 时 不 要 忘记 将 字符 编码 设置 为 UTF-8 
进行 保存 。 


























使 用 PHP 的 时 候 经 常会 出 现 不 按照 预期 运行 的 情况 。 拿 上 一 节 的 示例 来 说 ， 比 如 没有 显示 
“欢迎 光临 !” 等 。 这 一 定 是 由 某 些 原因 造成 的 ， 我 们 要 一 个 一 个 地 去 探究 。 
PHP 不 能 正常 运行 的 原因 主要 有 以 下 几 种 。 














P 拼写 错误 

如 果 运 行 失败 ， 最 大 的 原因 可 能 是 PHP 脚本 中 有 拼写 错误 ， 例 如 写成 zphp?、<php、prnt， 
或 者 没有 加 上 “9?”“"” 等 。 当 然 ， 除 了 “欢迎 光临 !” 以 外 ， 其 他 内 容 必须 使 用 半角 。 

在 大 多 数 情况 下 ， 错 误 信息 会 像 图 15-8 这 样 显 示 出 来 。 





























@ 名 httpy/localhosVlestLphp 只 - 6 | 恒 HTTP 500 内 部 服务 器 错 .x | 
0@ 网 站 无 法 品 示 该 页 面 
最 可 能 的 原 内 是 : 
。 该 网 站 正在 进行 维护 . 
”该 网 站 有 程序 庄 误 。 
你 可 以 宕 试 以 下 操作 : 
二” 恒 新 该 页 面 . 
旬 返回 双 上 一 页 。 


加 详细 信息 


图 15-8 错误 显示 画面 
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在 某 些 环境 下 可 能 会 出 现 即 使 发 生 了 错误 也 不 显示 错误 消息 的 情况 。 


输入 了 全 角 空 格 

在 程序 的 世界 中 ， 全 角 空 格 被 当 作 字符 使 用 。 和 在 SQL 语句 中 一 样 ， 字 符 必 须 用 " "或 ' ， 
括 起 来 。 也 就 是 说 ， 在 执行 缩 进 时 使 用 全 角 空 格 ， 或 者 在 关键 字 之 间 使 用 全 角 空 格 都 会 导致 错误 发 
生 。 只 要 使 用 中 文 ， 就 很 难 摆脱 全 角 空 格 的 问题 ， 我 们 要 多 加 小 心 。 









































> 不 是 可 以 使 用 PHP 的 环境 

这 是 可 以 使 用 PHP 的 环境 吗 ? 已 经 安装 好 PHP 了 吗 ? 这些 都 是 容易 忽略 的 错误 。 重 新 阅读 第 
2 章 ,， 确 认 一 下 设置 是 否 有 问题 。 

另外 ， 为 了 让 “欢迎 光临 !” 页 面 显示 在 浏览 器 上 ， 我 们 需要 确保 Apache 是 启动 状态 。 如 果 安 装 
了 MAMP ， 请 再 次 在 地 址 栏 中 输入 http:/localhosVMAMP ， 确 认 是 否 显示 出 和 15.7.1 节 一 样 的 内 容 。 

















PP 保存 在 不 同 的 文件 夹 中 
Apache 发 布 的 文件 夹 是 固定 的 。 但 是 ， 这 个 发 布 的 文件 夹 会 根据 环境 的 不 同 而 发 生变 化 。 文 
件 是 否 保存 在 了 正确 的 文件 夹 里 ?请 再 次 确认 (一 15.7.2 市 )。 























BP 扩展 名 不 是 “.php” 

要 发 布 的 PHP 文件 的 扩展 名 必须 是 “.php”。 那 么 ， 附 加 的 扩展 名 是 否 正确 ? 在 确认 扩展 名 之 
前 ， 务 必 将 扩展 名 设置 为 显示 状态 〈 一 2.2.1 节 )。 在 使 用 文本 编辑 器 的 情况 下 ， 文 件 名 可 能 会 变 为 
“test.php.txt”， 实 际 扩 展 名 为 “.txt”。 
































PP 字符 乱码 

如 果 是 本 书 介绍 的 环境 ， 乱 码 的 情况 应 该 不 会 发 生 ， 万 一 发 生 了 乱码 ( 见 图 15-9 )， 请 确认 以 
下 几 点 。 

如 上 所 述 ， 本 书 是 以 使 用 Unicode (UTF-8 ) 向 浏览 器 输出 为 前 提 进 行 介绍 的 。 请 确认 PHP 的 
程序 文件 是 否 使 用 了 UTF-8 保存 ， 如 果 是 GB 2312 等 字符 编码 ， 请 使 用 UTF-8 重新 保存 之 后 青 执 
行 。 另 外 ， 请 确认 php.ini 的 多 字 节 字符 的 相关 设置 。 

如 果 这 样 也 不 能 解决 问题 ， 请 纯 载 MAMP， 然 后 重新 安装 。 可 以 从 开始 按钮 中 选择 
MAMP 一 > Uninstall MAMP 来 仓 载 MAMP。 
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http://localhost/testphp PDO | localhost x tn 
绎 医 兰 适 莲 合 … 











15-9 乱码 的 示例 


15.8 使 用 PHP 


15.8.1 编写 PHP 脚本 时 需要 遵循 的 规则 
虽然 显示 的 内 容 只 有 一 行 ， 但 是 也 成 功 显 示 在 了 浏览 器 上 。 接 下 来 我 们 从 基础 开始 学 习 PHP。 
下 面 是 编写 PHP 脚本 时 需要 遵循 的 主要 规则 。 图 15-10 是 相应 的 示意 图 。 












































3 PHP 脚本 文件 的 扩展 名 是 “.php” 

3 PHP 脚本 以 “<?php” 开 始 ， 以 “?>” 结 束 
2 在 行 尾 加 上 “;” 

2 字符 串 数据 使 用 "" 或 '' 括 起 来 














<?Php 9 以 它 开始 











print "欢迎 光临 ! "; 最 


以 它 结束 





ol 
Wal 











?> 














图 15-10 PHP 的 基本 语法 
我 们 来 看 看 各 个 规则 的 具体 内 容 。 


Pp PHP 脚本 文件 的 扩展 名 是 “.php” 


PHP 脚本 文件 的 扩展 名 为 “.php”。 在 15.7.2 节 的 示例 中 ，PHP 脚本 文件 的 文件 名 为 test.php。 
虽然 可 以 设置 为 其 他 扩展 名 ,但 本 书 统一 使 用 “.php”。 


























> PHP 脚本 以 “<?php” 开 始 ， 以 “?>” 结 束 
在 扩展 名 为 “php” 的 文本 文件 中 编写 下 面 的 脚本 。 
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PHP 脚本 的 写法 


: <?php 


其 中 ,“<?php” 的 部 分 是 开始 标签 ,“? >” 的 部 分 是 结束 标签 。 也 就 是 说 ， 如 果 出 现 了 
“<?php”， 就 表示 PHP 代码 从 这 里 开始 ; 如 果 出 现 了 “?>”， 就 表示 PHP 代码 到 这 里 结束 。 
在 15.7.2 节 的 示例 中 ， 文 件 里 只 有 PHP 脚本 。 但 其 实 PHP 脚本 也 可 以 包含 在 HTML 文件 中 。 
即使 PHP 脚本 与 HTML 的 描述 摊 杂 在 一 起 ， 我 们 也 可 以 通过 “<?php” 和 “? >” 知 道 哪 部 分 是 
PHP 脚本 。 
PHP 脚本 可 以 多 次 记述 在 HTML 中 。 关 于 HTML 标签 的 描述 ， 我 们 将 在 17.3 节 进 行 介 绍 。 
































在 行 尾 加 上 “? 
行 的 末尾 要 加 上 “;”( 分 号 )。 当 错误 发 生 时 ， 首 先 要 确认 一 下 是 否 是 这 里 出 现 了 问题 。 但 如 
果 是 testphp 这 种 只 有 一 行 脚本 的 文件 ， 即 使 不 加 上 “;” 也 能 正常 运行 。 


> 字符 串 数据 使 用 " "或 '' 括 起 来 

字符 串 数 据 需 要 使 用 " "或!' 括 起 来 。 也 就 是 说 ， 在 脚本 中 不 允许 出 现 不 用 " "或 '' 括 
起 来 的 字符 串 。 

另外 , 在 PHP 中 用 " " 括 起 来 的 变量 能 够 被 解析 , 但 如 果 用 ' “' 括 起 来 就 不 能 被 解析 了 “。 把 
变量 括 起 来 时 使 用 " " 和 使 用 ' “的 区 别 会 在 16.2.2 节 介 绍 。 








编写 PHP 脚本 
本 书 使 用 了 “<?php” 和 “?>” 来 编写 PHP 脚本 ， 不 过 我 们 也 可 以 通过 修改 php.ini 的 设置 ?来 使 用 “<2” 
和 “?>” 编 写 。 以 前 的 版 本 也 允许 使 用 “<s” 和 “#>"”， 但 是 这 个 标签 在 PHP 7 中 已 经 不 能 使 用 了 。 
















































































六 15.8.2 执行 了 什么 处 理 


在 上 一 节 的 PHP 脚本 文件 testphp 中 执行 “print " 欢迎 光临 1"”， 浏 览 絮 就 会 显示 出 “欢迎 
光临 !” 的 字样 。 我 们 再 来 回顾 一 下 这 个 机 制 ( 见 图 15-11 )。 


























(DD 在 PHP 中， 双 引 号 中 的 变量 ($var ) 和 特殊 字符 (\r\n 等 ) 会 被 转 义 ， 而 单 引 号 中 的 内 容 总 被 认为 是 普通 字 
符 ， 因 此 不 会 被 转 义 。 译 者 注 
@ 设置 参数 为 short_open tag 和 asp_tags (Removed from PHP 7.0 )。 








译 者 注 
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区 访问 test.php 文件 
首先 ， 在 浏览 器 的 地 址 栏 中 输入 http:// x x x x/test.php， 访问 test.php 文件 。 收 到 请 求 的 Web 
服务 需 会 执行 test.php。 





”Web 服务 器 进行 处 理 并 返回 结果 

在 15.7.2 节 的 示例 中 ， 实 际 运 行 的 是 “print “" 欢迎 光临 !" ;” 的 部 分 。print 是 用 于 显示 
字符 串 的 命令 。 也 就 是 说 ， 收 到 这 个 命令 的 Web 服务 器 将 字符 串 “ 欢 迎 光临 !” 发 送 到 客户 端的 浏 
览 器 。 于 是 ,浏览 器 便 显示 出 了 “欢迎 光临 1”。 


< 和 < oT 


15-11 执行 的 处 理 












































P 插入 HTML 标签 

要 想 显 示 字 符 串 ， 就 需要 使 用 print，HTML 标签 也 和 字符 串 一 样 可 以 通过 print 显 
示 出 来 。 

例如 ，<br> 是 表示 换行 符 的 HTML 标签 。 代 码 清单 15-2 在 字符 串 " 欢迎 光临 " 之 后 ， 通 过 
HTML 标签 <br> 进行 换行 ， 然 后 显示 "SQL 咖啡 厅 !"。 











代码 清单 15-2 br.php 





<?php 

print " 欢迎 光临 "; 
Brint Vebrs ry 

print "SQL 咖啡 厅 ! "; 


人 














执行 上 述 人 代码， 浏览 右 中 就 会 显示 图 15-12 的 内 容 。 
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httpy/localhostybrphp 








€ © localhostbrphp 


欢迎 光临 
SQL 咖啡 厅 ! 





图 15-12 br.php 的 执行 示例 
显示 源 代码 (一 17.1 节 ) 可 以 看 到 如 下 内 容 。 


欢迎 光临 <bz>sez 咖 啡 厅 ! 


print 和 echo 

和 print 功能 类 似 的 命令 还 有 echo。 二 者 的 使 用 方法 相同 ， 但 是 print 会 返回 TRUE 这 个 值 (TRUE 被 视 
为 1)， 而 echo 不 返回 值 。 

我 们 来 看 一 个 例子 。 当 执行 以 下 代码 时 ， 浏 览 器 上 会 显示 什么 内 容 呢 ? 
































Pprintee erin he 


<?php 
print print "你 好 "; 


-> 








这 时 ， 浏 览 器 上 会 显示 “你 好 1”"。 这 是 因为 右边 的 “print "你 好 " ”输出 了 “你 好 ”"， 左 边 的 print 输 
出 了 它 的 返回 值 TRUE ( 1 )。 
而 执行 下 面 的 代码 会 发 生 错 误 。 





























Pecho echo.php 


<?php 
echo echo "你 好 "; 

















因为 echo 无 法 输出 没有 返回 值 的 echo， 连 续 使 用 两 个 echo 是 错误 的 ， 所 以 代码 发 生 了 错误 。 这 么 想 就 容 
易 理 解 了 。 另 外 , “pzrint echo "你 好 ";”( 示例“print_echo.php”) 也 会 报错 ， 而 “echo print "你 好 ";” 
( 示例 echo_print.php ) 则 不 会 报错 ， 会 显示 出 “你 好 1”。 

本 书 将 统一 使 用 print。 
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C 15.8.3 注释 的 写法 


我 们 可 以 在 脚本 中 按照 自己 的 习惯 编写 备忘录 。 这 个 备忘录 称 为 注释 。 当 执行 程序 时 ， 注 释 则 
分 会 被 忽略 掉 ， 所 以 我 们 可 以 在 注释 部 分 编写 任何 内 容 。 

时 间 一 长 ， 创 建 程序 的 人 也 会 忘记 编写 的 内 容 。 所 以 最 好 把 哪 部 分 代码 执行 了 哪些 处 理 等 重点 
内 容 作 为 备忘录 写 下 来 。 另 外 ， 当 程序 不 能 正常 运行 时 ， 可 以 通过 注释 将 可 疑 的 代码 注释 掉 使 其 停 





























止 运行 ， 这 样 也 比较 容易 找 出 问题 出 现 的 原因 。 


注释 的 编写 规则 妇 


2 如 果 代 码 以 “/ 
2 如 果 代 码 在 “/ 


[下 所 述 。 


/” 或 “#” 开 头 ， 则 该 行 不 会 执行 任何 操作 
*” 和 “*/” 之 间 ， 则 该 部 分 不 执行 任何 操作 


代码 清单 15-3 是 包含 注释 的 示例 。 这 段 代码 和 15.7.2 节 的 示例 一 样 仅 执行 “print "欢迎 光 














代码 清单 15-3 comment 





年 ! "7 。 我 们 也 可 以 使 用 “/” 或 “*” 使 注释 变 得 更 加 醒目 。 





.php 





<?php 
//” 这 一 行 会 被 全 部 





注释 掉 








/* 

可 以 包括 

多 行 的 

注释 

#Y 

print " 欢迎 光临 ! 





?> 





# ”这 个 符号 也 表示 注 


/炎炎 炎炎 火炎 炎炎 炎炎 炎炎 炎炎 火灾 灾 灾 火炎 类 这 是 一 条 引 人 注 的 注释 六 炎炎 炎炎 炎炎 炎炎 类 类 类 类 交 交 类 大大 类 类 类/ 


/1/11/11111111111111111111/ ”这 条 注释 也 很 显眼 ffVV11111111111111111111 


E 释 





";/* 在 行 中 间 输 入 注释 */ 















































S 15.8.4 ”phpinfo 函数 
下 面试 着 使 用 PHP 的 函数 。PHP 中 也 有 大 量 函 数 。 处 理 内 容 涵盖 字符 串 、 数 字 、 文 件 和 数据 














库 操作 、 网 络 等 ， 数 量 庞 大 到 让 人 觉得 所 有 处 理 都 已 经 作为 函数 准备 好 了 。 





我 们 会 在 之 后 的 章节 中 详细 介绍 函数 ， 这 里 先 试 着 使 用 显示 PHP 环境 信息 的 phpinfo 函数 。 
代码 清单 15-4 是 执行 pnpinfo 函数 的 PHP 脚本 文件 。 





代码 清单 15-4 info.php 





<?php 
BPRpinfo()y 


全 
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PHP 函数 后 面 的 () 中 要 写 上 传递 给 函数 的 参数 (一 8.2.2 节 )。 在 这 种 情况 下 ， 即 使 函数 没有 
参数 也 必须 加 上 () 。 这 一 点 与 MySQL 的 函数 相同 。 

请 将 执行 phpinfo 函数 的 PHP 脚本 文件 命名 为 info.php 并 保存 在 发 布 的 目录 中 。 在 地 址 栏 中 
输入 http://localhost/info.php， 就 会 显示 图 15-13 中 的 内 容 。 
























































过 口 x 
) 回 http://localhost/info.php PC | 回 pnpine0 x| 介 六 次 图 
System Windows NT DESKTOP-8AAOODU 6.2 build 9200 (Windows 8 Home Premium Edition) i586 
Build Date May 19 2017 10:18:29 
Compiler MSVC14 (Visual C++ 2015) 
Architecture X86 
Configure Command cscript /nologo configure js ~ 一 disable-alr "enable-cli" "—enable-cgi" -enable-apache2-: ee “enable- 
apache2-4handier "—with-bz2=shared" "—enable-mbstring=shared" "—enable-exif=shared" 
wi "with-sqlite3=shared" "—with-curf=shar ee "with-gd=shared" 
enablejson=static” ion=shared" "with-mysqli=shared, C:\MAMP\bin\mysql” 
mcrypt=shared" "—with-mysqind=shared” *- pcache=shared" "--enable-apcu=shared" 
lable-ftp=shs i, lared" "—with-gmp=shared" "with- 一 enable-int=shared" "一 
with-ldap=shared" "—with-pgsql=shared" " ebird=shared" "—with-pdo-odbc=shared" "—with-i 
pgsql=shared" “enable-shmop=shared" "—enable-phar" "一 enable-hash” 
Ea “with-pdo-mysqr "一 with-xmr "—enable-ziib” "—enable-zip" "—enable-ctype” "enable- 
fiter’ "—enable-sockets" "—enable-calendar”" "enable-bcemath” "enable-fil " "—enable-xmlreader" "--enable- 
Xml 一 With- " "~enabl et” bc” ~- 
mbregex" "with-imagi red, C:\prog\mageMagick-6.8.9;C:\prog\ImageMagick-6.8.9\VisualMagick\lib" "—with- 
xs 
ServerAPI Apache 2.0 Handler 
Virtual Directory Support enabled 
Configuration File (php.ini) Path CWINDOWS 
Loaded Configuration File CMAMP\conf\php7.-1.5\php.-ini 
Scan this dir for additional .ini files (none) 
Additional .ini files parsed (none) 
PHP API 20160303 
PHP Extension 20160303 
Zend Extension 320160303 
Zend Extension Build API320160303,TS,VC14 MY 




















15-13 ”执行 结果 ( http://localhost/info.php ) 

从 显示 的 内 容 来 看 ， 除 了 PHP 的 版 本 和 当前 的 设置 内 容 之 外 ,PDO (一 18.1.2 节 ) 驱动 的 相 
关 信 息 、 通 过 Apache 设置 的 环境 变量 的 相关 内 容 也 显示 了 出 来 。 虽 然 本 书 不 会 涉及 具体 细节 ,但 
后 面 会 介绍 REMOTE_ADDR 等 环境 变量 的 相关 内 容 (一 16.3.3 节 )。 

如 果 PHP 或 MySQL 没有 按照 预期 运行 ， 我们 也 可 以 执行 该 函数 来 检查 相关 设置 。 男 外 ， 
phpinfo 困 数 也 可 以 通过 从 MAMP 的 开始 页 (http://localhost/MAMP/ ) 选择 画面 上 部 的 “phpinfo” 
来 执行 。 

















使 用 PHP 关闭 操作 系统 

PHP 中 提供 了 很 多 函数 。 下 面 就 来 介绍 一 下 其 中 一 个 比较 有 趣 的 函数 。 

执行 命令 的 函数 中 有 exec 这 个 函数 。exec 会 执行 参数 中 指定 的 命令 。 也 就 是 说 ，exec 函数 可 以 代替 通 
过 命令 提示 符 和 终端 输入 的 命令 使 用 。( 但 是 ， 并 不 是 所 有 的 命令 都 能 执行 。) 

关闭 操作 系统 的 命令 是 SHUTDOWN -s。 也 就 是 说 ,创建 一 个 内 容 为 exec ("SHUTDOWN -s ") 的 PHP 脚 
本 ， 然 后 执行 ，1 分 钟 后 操作 系统 就 会 关闭 。 我 们 可 以 使 用 “SsHUTDOWN -s -t 10”( 10 秒 后 关机 ) 指定 关机 
之 前 的 等 待 时 间 。 

如 果 在 localhost 上 运行 代码 清单 15-5 的 PHP 脚本 ， 计 算 机 就 会 在 10 秒 后 关闭 。 




































































338 | 第 15 章 用 于 控制 MySQL 的 PHP 


代码 清单 15-5 ”shutdown.php 





<?php 
exec ("SHUTDOWN -s -t 10"); 


?> 




















另外 ，PHP 会 在 服务 器 端 ( 本 书 中 是 localhost ) 执行 。 也 就 是 说 关闭 的 是 服务 器 端 ， 干 万 不 要 弄 错 。 


本 章 介 绍 了 以 下 内 容 。 





@ Apache、MySQL、PHP 的 动作 和 它们 在 Web 应 用 程序 的 执行 中 所 起 到 的 作用 
@ 使 用 PHP 的 步骤 

@ 编 写 PHP 脚本 的 基础 知识 

@ 运 行 PHP 脚本 的 方法 


> 自我 检查 
下 面 检查 一 下 本 童 学 习 的 内 容 是 否 全 部 理解 并 掌握 了 。 

















口 能 够 理解 Web 服务 器 的 作用 

口 知道 编写 PHP 脚本 的 基本 规则 
口 能 够 运行 PHP 脚本 

口 能 够 创建 输出 字符 串 的 PHP 脚本 


































































































下 面 是 具有 “在 经 过 指定 的 秒 数 后 自动 跳 转 到 指定 URL” 功 能 的 标签 。 请 使 用 这 个 标签 ， 创 建 能 够 在 
5 秒 后 自动 跳 转 到 Web 页 面 http://www.boc.cn/ 的 PHP 脚本 。 
































国 <meta http-equiv='refresh'> 标签 的 功能 









































使 用 PHP 脚本 编写 标签 即 可 。 














创建 代码 清单 15-6 的 PHP 脚本 。 


代码 清单 15-6 示例 move.php 





<?php 


URL= http://www.boc.cn/'>"; 








U 
































print "5 秒 后 跳 转 到 中 国 银行 的 页 


?> 





print "<meta http-equiv='refresh' 


1, 
’ 


Contents!'Ds 
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在 第 15 章 中 ， 我 们 使 用 PHP 将 “欢迎 光临 !” 的 字样 显示 
在 了 浏览 器 上 。 本 章 我 们 将 学 习 PHP 的 基础 知识 。 

PHP 很 深奥 ， 光 函数 就 有 1000 多 个 。 在 本 书 中 ， 我 们 仅 讨 
论 MySQL 使 用 方面 的 PHP 知识， 主要 介绍 控制 MySQL 所 需要 
的 变量 、 字 符 串 、 函 数 、 比 较 运 算 符 、 条 件 判 断 、 循 环 和 数组 。 








忆 | 










































































C 16.1.1 什么 是 PHP 的 变量 


首先 来 看 变量 。 我 们 介绍 过 变量 是 一 个 用 于 保存 值 的 “箱子 ”( 一 12.5.3 节 )。 变 量 在 PHP 与 
MySQL 的 结合 方面 发 挥 着 不 可 替代 的 重要 作用 。 
beth sy 首先 按照 15.7.2 节 的 方法 输入 代码 清单 16-1 的 脚本 。 


代码 清单 16-1 variable.php 



































<?php 
$a= " 欢迎 光临 1"; 
print $a; 


?> 














确认 无 误 后 ， 将 该 文件 命名 为 variable.php 并 保存 到 Web 服务 器 发 布 的 文件 夹 中 。 不 要 忘记 保 
存 时 将 字符 编码 设置 为 UTF-8。 不 同 的 Web 服务 器 发 布 的 文件 夹 也 有 所 不 同 ， 如 果 像 本 书 一 样 安装 
了 MAMP， 文 件 夹 就 是 CNMAMP\htdocs (一 15.7.2 节 )。 

确认 Apache 正在 运行 (一 2.2.3 节 ) 后 ,在 Web 浏览 需 的 地 址 栏 中 输入 以 下 内 容 。 














http://localhost/variable.php 














这 样 就 会 显示 出 图 16-1 的 页 面 。 





http://localhost/variable.php x 


© | | Q search 


(€ 了】 © localhost/variable.php 


El 
变量 | 
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| 人 女 自 咏 > 








欢迎 光临 ! 


16-1 执行 结果 
如 果 没 有 正常 显示 ， 请 参照 15.7.3 节 的 内 容 进行 检查 。 








下 面 我 们 来 看 一 下 脚本 的 内 容 。 首 先 ，PHP 在 变量 名 的 开头 加 了 $。 上 上述 示例 中 的 变量 是 a， 


于 是 就 变 成 了 $a。 重 点 是 下 面 这 个 表达 式 。 


$a=" 欢 迎 光临 1!" ; 











初次 接触 编程 语言 的 读者 ， 看 到 这 个 表达 式 也 许 会 觉得 不 可 思议 。 这 个 表达 式 表示 将 字符 串 








“欢迎 光临 !” 赋 给 变量 





Sao 











在 程序 的 世界 中 ,“=” 表 示 将 = 右边 的 值 赋 给 左边 。 也 就 是 说 ，$a 的 内 容 是 


( 见 图 16-2 )。 


变量 


16-2 


另外 , 在 MySQL 的 比较 运算 符 中 出 现 的 表示 相等 的 “ 


全- 二 对 (一 16.4 节 )。 


“欢迎 光临 1” 


=”， 在 PHP 中 需要 连续 写 两 次 ， 即 








如 上 所 述 ，print 是 用 于 输出 字符 串 的 命令 。 因 此 ， 下 国 





i 的 命令 会 输出 $a 的 值 ， 即 输出 字符 





串 


“欢迎 光临 1”。 
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其 实 我 们 只 是 使 用 变量 重新 表述 了 一 下 15.7.2 节 的 示例 。 这 种 用 于 保存 值 的 “箱子 ”就 是 变 
量 。 另 外 ， 值 并 不 是 一 直 保存 在 变量 中 的 ， 我 们 也 可 以 自由 地 对 其 进行 存 取 。 





| 中 




















16.1.2 ”变量 名 的 规则 
设置 变量 名 需要 遵循 各 种 规则 。 在 PHP 的 情况 下 ， 有 如 下 规则 。 


2 以 “$” 开 头 
DS 对 大 小 写 敏 感 
2 由 字母 、 数 字 和 “ ”( 下 划 线 ) 组 成 
3“$” 后 面 不 能 以 数字 开头 
P 以 “$” 开 头 


PHP 变量 的 特点 是 在 开头 加 上 前 级 “$”。 例 如 $age、S$TEL 等 。 
P> 由 字母 、 数 字 和 “”( 下 划 线 ) 组 成 ， 但 不 能 以 数字 开头 


变量 的 开头 必须 是 字母 或 者 ““”。 像 S30years 这 样 的 写法 是 不 允许 出 现 的 , 但 $_30 
years 就 没有 问题 。 





> 对 大 小 写 敏 感 


对 大 小 写 敏感 ， 举 例 来 说 就 是 变量 $tel、$Tel 和 $TEL 各 不 相同 。 我 们 在 编写 脚本 的 时 候 会 
不 知 不 觉 混合 使 用 $x 和 $x， 一定 要 注意 字母 的 大 小 写 。 本 书 中 所 有 的 变量 名 都 使 用 了 小 写字 母 。 


























S 16.1.3 ”预定 义 常量 


在 PHP 中 ， 有 些 值 即使 用 户 不 设置 也 会 被 事先 定义 好 ， 这 种 类 型 的 值 就 是 预定 义 常 量 。 例 如 ， 
表示 圆周 率 和 PHP 版 本 的 常量 就 可 以 被 直接 使 用 。 








> PHP 常量 的 示例 
M_PI 昼 周 率 
PHP_VERSION PHP 的 版 本 





























PHP_OS 运行 的 OS 














执行 代码 清单 16-2 中 的 代码 ， 圆 周 率 就 会 显示 出 来 ， 如 图 16-3 所 示 。 





代码 清单 16-2 pi.php 





<?php 
print M PIS 


?> 























与 ) 儿 c) 园 http://localhostpiphp Pro | 园 ecanost X | 从 妆 这 © 


3.1415926535898 














图 16-3 ”执行 结果 


16.1.4 变量 的 数据 类 型 

下 面 来 看 变量 的 数据 类 型 。PHP 具有 即使 不 定义 数据 类 型 也 可 以 使 用 变量 的 特征 。 赋 值 时 PHP 
会 自动 决定 数据 类 型 ， 值 为 字符 串 就 是 字符 串 类 型 ， 值 为 整数 就 是 整数 类 型 ， 不 需要 进行 定义 。 

当 使 用 MySQL 创建 表 时 ， 必 须 一 开始 就 决定 列 的 数据 类 型 ( 一 5.1 节 )， 因 此 基本 不 可 能 在 数 
值 类 型 的 列 中 写 和 字符 串 。 此 外 ， 对 于 存储 函数 中 使 用 的 变量 ， 我 们 必须 事先 使 用 DECLRRB 来 声 
明 类 型 (一 12.5.3 节 )。 

PHP 相对 来 说 比较 灵活 ， 它 会 根据 输入 的 值 决定 相应 的 数据 类 型 。 

表 16-1 列 出 了 几 个 能 够 在 PHP 中 使 用 的 数据 类 型 。 
























































表 16-1 能够 在 PHP 中 使 用 的 数据 类 型 


























内 容 数据 类 型 

整数 integer 

浮 点 数 float、 double 
字符 串 string 

布尔 值 boolean 

对 象 object 

数组 array 

资源 resource 

空 值 NULL 

















赋值 之 后 ，PHP 会 自动 设置 相应 的 数据 类 型 。 目 前 我 们 不 用 在 意 这 方面 的 内 容 。 
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16.2 ”字符 串 


下 面 将 介绍 PHP 字符 串 的 相关 内 容 。 














人 16.2.1 连接 字符 惠 
我 们 来 看 一 下 连接 字符 串 的 方法 。 在 PHP 中 使 用 “.” 连 接 字符 串 。 字 符 串 数据 需要 用 ， ， 
或 ，， 括 起 来 ， 因 此 在 连接 “ 西 泽 ”和 “ 梦 路 ”这 两 个 字符 串 时， 需要 写成 “" 西 泽 "." 梦 路 ""。 
代码 清单 16-3 将 “你 好 ” 赋 给 变量 $a， 将 “欢迎 光临 sor 咖啡 厅 !” 赋 给 变量 $b， 然 后 连接 
这 两 个 字符 串 并 显示 出 来 。 


代码 清单 16-3 join.php 


















































<?php 

$a= "你 好 "; 

$b=" 欢迎 光临 SQz 咖啡 厅 !1"; 

printe Sa 


?> 











$a. $b 将 各 个 变量 中 的 字符 串 连 接 了 起 来 ( 国 )。 


C 16.2.2 “"” 和 “" ”的 使 用 方法 

字符 串 需要 用 " "或 '' 括 起 来 。 

但 是 ,使 用 " " 括 起 来 的 字符 中 不 能 包括 “"”,， 使 用 ， “，' 括 起 来 的 字符 中 不 能 包括 “'”。 这 
是 因为 如 果 有 “"” 或 “'”，PHP 会 认为 它 是 字符 串 的 起 点 或 终点 ， 导 致 错误 发 生 。 这 一 点 和 
MySQL 的 SQL 语句 是 一 样 的 。 

例如 ， 执 行 下 面 的 命令 会 发 生 错 误 。 























Print "将 字符 串 用 "" 括 起 来 "; 

















P 转 义 处 理 
在 这 种 情况 下 ， 我 们 可 以 通过 在 “" ”或 “' ”之 前 加 上 从 ”来 避免 发 生 错误 ， 这 种 处 理 称 为 
转 义 处 理 (一 5.3.2 节 )( 见 图 16-4 )。 
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print " 将 字符 串 用 \wNn" 括 起 来 "; 


























“"” 的 前 面 要 加 上 "\”( 转 义 处 理 ) 























图 16-4 转 义 处 理 


使 用 "" 或” 
还 有 另 一 种 方法 ， 那 就 是 当 “"” 需 要 作为 字符 处 理 时 ,用 ' “' 括 起 来 ， 或 者 当 “' ”需要 作 
为 字符 处 理 时 ,用 " " 括 起 来 。 比 如 下 面 这 这 种 写法 。 

















print "将 字符 串 用 ' ' 括 起 来 "; 





在 以 后 的 章节 中 ， 当 使 用 MySQL 创建 Web 应 用 程序 时 ， 我 们 会 通过 PHP 脚本 执行 SQL 语 
句 。 ee SQL 语句 会 作为 字符 串 指 定 给 query 方法 的 参数 (一 18.2 节 )。 
会 详细 介绍 query 和 方法 的 相关 内 容 ， 现 阶段 我 们 只 要 记 住 query ( 作为 字符 串 的 SQL 
ee ts 
那么 当 执 行 SQL 语句 INSERT INTO tbl VALUES ('A101'，,...) 时 该 如 何 描述 呢 ? 
SQL 语句 是 字符 串 数据 ， 需 要 用 ' “' 括 起 来 ， 但 使 用 ' “将 包含 ' ， 的 8 SQL 语句 括 起 来 时 会 发 


生 错 误 。 





























Guery (! INSERT INTO tbl1 VALUES('A101',...)') 














使 用 ' ' 将 ' ' 括 起 来 的 话 会 报错 


这 时 就 需要 采取 一 些 方 法 来 解决 这 个 问题 了 ， 比 如 用 “\ ”进行 转 义 ,或 者 像 下 面 这 样 在 外 部 
使 用 " "， 从 而 让 内 部 的 “' ”作为 字符 来 处 理 。 












































query ("INSERT INTO tbl1 VALUES('A101',...)") 

















使 用 " "将 ' ' 括 起 来 的 话 不 会 出 现 问 题 


或 者 可 以 在 外 部 使 用 '， 从 而 将 内 部 的 “"” we 

当 使 用 PHP + MySQL 时 ， 首 先 会 遇 到 的 难题 就 是 “'” 机 以 后 我 们 会 学 习 
query 方法 的 相关 内 容 ， 这 里 先 记 住 不 可 以 使 用 ， ee ” 括 起 来 ,或 者 使 用 " " 
将 作为 字符 的 “"” 括 起 来 。 



































16.2.3 用 "" 将 变量 括 起 来 和 用 '' 将 变量 括 起 来 的 区 别 


不 管 是 " "还 是 ' '， 都 可 以 将 字符 串 数 据 括 起 来 ， 但 是 在 变量 的 情况 下 ， 二 者 的 处 理 结果 
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在 代码 清单 16-4 中 ，" "中 的 变量 会 被 解析 ， 执 行 结果 中 会 显示 变量 的 值 “123”( 国 )， 如 
图 16-5 所 示 。 


代码 清单 16-4 double_quotation.php 





<?php 
$a=123，; 
eele "WSep 


?> 




















一 口 Xx 
(年 ) 晤 ) 国 mapwocahosvdoubeqa P ~ OIocalhost x 国 全 交 瘟 @ 














图 16-5 ”执行 结果 
但 在 代码 清单 16-5 的 情况 下 ，' “' 中 的 变量 就 不 会 被 解析 ， 而 是 当成 字符 串 来 处 理 。 也 就 是 
说 ，s$a 会 作为 字符 串 显示 出 来 ( 见 图 16-6 )。 


代码 清单 16-5 ”single_quotation.php 











<?php 
$a=123，; 
Sueulate haw 


?> 























Ce 外 只 )| 园 httpi//ocalhost/singe qu DY 0 | 回 localhost x | | 价 交 产 辐 














16-6 ”执行 结果 
使 用 PHP 执行 的 SQL 语句 中 如 果 存 在 变量 ， 就 需要 我 们 多 加 注意 了 。 
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16.3 ”函数 


关于 MySQL 的 函数 ， 我 们 已 经 在 8.2.2 节 中 进行 了 介绍 。PHP 中 也 有 函数 。 


S 16.3.1 本 书 涉及 的 PHP 函数 
PHP 函数 有 1000 多 个 ， 表 16-2 只 列 出 了 本 书 出 现 的 函数 。 


表 16-2 本 书 出 现 的 函数 


































































































函数 名 内 容 

date 返回 当前 日 期 和 时 间 ( 一 16.3.2 节 ) 

exec 执行 命令 ( 一 15.8.4 节 ) 

phpinfo 显示 PHP 的 信息 (一 16.3.3 节 ) 

nl12br 在 需要 换行 的 情况 下 ， 插 入 HTML 换行 标签 (一 17.7.2 节 ) 
preg_ match 使 用 正则 表达 式 执行 模糊 查询 ( 一 20.3.2 节 ) 
htmlspecialchars 转换 标签 等 特殊 字符 串 (一 20.4.3 节 ) 
isset 仿 查 是 否 设 置 了 变量 ( 一 21.3.1 节 ) 

getenv 获取 环境 变量 ( 一 16.3.3 节 ) 
gethostbyname 通过 主机 名 获取 IP 地 址 ( 参考 ) 
gethostbyaddr 通过 IP 地 址 获取 主机 名 (一 16.3.1 节 ) 








* 


下 面 以 date、phpinfo、getenv 和 gethostbyaddr 涵 数 为 例 进行 介绍 。 


S 16.3.2 ”通过 date 函数 显示 日 期 和 时 间 


首先 我 们 来 介绍 日 期 和 时 间 处 理 中 必 不 可 少 的 aate 函数 。 当 使 用 这 种 与 日 期 和 时 间 相关 的 函 
数 时 ， 必 须 事先 正确 地 设置 好 时 区 (一 15.6.3 节 )。 


date 函数 的 格式 











date 是 用 于 返回 时 间 的 函数 。 在 参数 中 指定 表 16-3 中 的 字符 串 ， 就 会 返回 相应 的 日 期 和 
时 间 。 
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表 16-3 date 函数 中 指定 的 字符 串 


时 间 的 格式 返回 值 


















































































































































g 12 小 时 制 的 小 时 

h 2 位 数字 表示 的 12 小 时 制 的 小 时 

如 24 小 时 制 的 小 时 

H 2 位 数字 表示 的 24 小 时 制 的 小 时 

j 期 

1 星期 的 英文 字符 串 ( 返回 Saturday 等 字符 ) 
F 月 份 的 名 称 ( 返回 January 等 字符 ) 
n 月 份 

m 2 位 数字 表示 的 月 份 " 

s 秒 (2 位 数 ) 

Y 年 份 

y 2 位 数字 表示 的 年 份 

















例如 ， 函 数 


date ("F") 





将 返回 December 等 当前 月 份 的 名 称 。 男 外 ， 隐 数 


date("n") 
date("j") 








将 返回 当前 的 月 份 和 日 期 。 如 果 需 要 在 Web 页 面 上 自动 显示 “x 年 x 月 x 日 "， 该 函数 会 起 到 很 
大 的 作用 。 另 外 ,我 们 可 以 在 date 函数 的 第 2 个 参数 中 于 3 日 期 和 时 间 (时 间 蕉 ) >。 

在 PHP 中 ， 日 期 和 时 间 的 信息 以 “时 间 戳 ”的 格式 来 处 理 。 这 个 “时 间 戳 ”是 从 1970 年 1 月 
1 日 0 点 起 到 现在 经 过 的 秒 数 。 











P 使 用 date 函数 

下 面 来 创建 显示 当前 日 期 的 脚本 。 如 代码 清单 16-6 所 示 ， 我 们 要 创建 将 当前 日 期 显示 为 “ 今 
天 是 x 年 x 月 x 日 ”的 脚本 。 和 前 面 介 绍 的 一 样 ， 使 用 “.” 连 接 字 符 串 。 执 行 结果 如 图 16-7 
所 示 。 






































@ 日 期 的 格式 实际 为 string date ( string S$format [，int $timestamp = time() ] ), 其 中 第 2 
个 参数 默认 为 当前 时 间 和 日 期 。 译 者 注 





译 者 注 





16.3 函数 | 349 


代码 清单 16-6 date.php 





<?php 
print T 今天 是 nm .date ("Y") A 年 LL .date ("m") | 月 Tm .date ("j") 。 日 mm 


?> 











http://localhost/date.php 


今天 是 2018 年 08 月 19 日 





16-7 ”执行 结果 


S 16.3.3 ”环境 信息 
P 通过 phpinfo 函数 获取 环境 信息 


在 用 于 获取 PHP 相关 信息 的 函数 中 ， 有 一 个 phpinfo 函数 ， 只 要 按照 下 面 的 方式 描述 ， 相 关 
信息 就 会 显示 出 来 (一 15.8.4 节 )。 





即使 函数 不 需要 参数 也 要 加 上 () (一 8.2.3 市 ),。 仅 通 过 “phpinfo() ;” 这 一 行 代码 就 能 
示 出 许多 环境 信息 。 

phpinfo 返回 的 信息 中 也 包含 环境 变量 。 这 些 信息 将 显示 在 phpinfo 消 数 输出 的 “Apache 
Environment” 的 模块 中 。 环 境 变 量 中 保存 着 Web 服务 絮 的 软件 和 客户 端的 IP 地址 等 重要 信息 。 在 
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REMOTE ADDR 项 目 中 应 该 可 以 看 到 机 器 的 卫 地 址 ， 大 家 可 以 去 确认 一 下 。( 在 localhost 的 情况 
下 是 “127.0.0.1” 之 类 的 全 地 址 。) 





> getenyv 函数 
下 面试 着 使 用 getenv 函数 创建 用 于 显示 客户 端 信息 的 PHP 























getenv 函数 是 用 于 返回 “环境 变量 的 值 ”的 函数 ， 我 们 只 定 特 定 的 参数 就 可 以 获取 相应 
的 信息 。 表 16-4 显示 了 getenv 也 we， 
getenv 函数 的 格式 


表 16-4 getenv 函数 中 指定 的 参数 和 获取 的 信息 


















































参数 ( 想 获取 的 信息 项 ) 获取 的 信息 
SERVER_SOFTWRARE Web 服务 器 软件 
SERVER_PORT 使 用 的 站 

PATH 服务 器 中 设置 的 路 径 
REMOTE ADDR 客户 端的 IP 地 址 
HTTP_USER AGENT 客户 端的 浏览 器 信息 




















比如 ， 我 们 可 以 通过 getenv ("SERVER _SOFTWARE") 获取 Web 服务 器 软件 的 信息 。 下 面 执 
行 代码 清单 16-7 中 的 代码 。 执 行 结果 如 图 16-8 所 示 。 


代码 清单 16-7 serv_disp.php 





<?php 
print getenv ("SERVER SOFTWARE"); 


Ee 














- OO x 
介 广 齐全 


Apache/2. 2.31 (Win32) DAV/2 mod_ ssl/2.2.31 OpenSSL/1.0.2e mod fcgid/2.3.9 
mod wsgi/3.4 Python/2.7.6 PHP/7.0.9 mod_ perl/2.0.8 Perlv5.16.3 


所 有 儿 c )) 回 hitp://localhost/serv disp DY 0 | 回 localhost X 


























16-8 ”执行 结果 
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同样 ， 我 们 也 可 以 通过 getenv ("REMOTE_ADDR") (示例 ip.php ) 获取 客户 端的 IP 地 址 。 执 
行 结果 如 图 16-9 所 示 。 

















代码 清单 16-8 ip.php 





<?php 
print getenv ("REMOTE ADDR"); 


Es 




















x 
人 《<s 儿 只 ) 园 http://ocalhostip.php D7 cd 国 Iocalhost x| 介 站 远 @ 


127.0.0.1 











图 16-9 执行 结果 


S 16.3.4 通过 gethostbyaddr 函数 获取 主机 名 


我 们 来 使 用 一 下 gethostbyaddr 图 数 。gethostbyaddat 函数 是 一 个 用 于 通过 IP 地 址 获取 
主机 名 的 函数 。 








gethostbyaddr 函数 的 格式 





只 要 将 上 节 中 通过 getenv ("REMOTE _ADDR") 获取 的 全 地址 作为 参数 传递 给 gethostbyaddr 
函数 ， 就 可 以 获取 客户 端的 主机 名 。 
代码 清单 16-9 是 用 于 显示 客户 端 主机 名 的 脚本 。 执 行 结果 如 图 16-10 所 示 。 


代码 清单 16-9 host_disp.php 











<?php 
print gethostbyaddr (getenv ("REMOTE ADDR")); 


?> 
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EN 回 http://localhost/host disp PD” 0 | 园 localhost x 





DESKTOP-8AAOODU 











16-10 ”执行 结果 


P 使 用 gethostbyaddr 函数 

我 们 来 创建 一 个 使 用 getenv 函数 和 gethostbyaddr 函数 返回 客户 端 信息 的 PHP 脚本 。 

这 个 脚本 中 使 用 了 getenv 函数 获取 客户 端的 卫 地 址 和 客户 端的 浏览 器 信息 ， 然 后 让 通过 
getenv 国 数 获取 的 IP 地 址 作为 gethostbyaddr 函数 的 参数 ， 从 而 获取 客户 端的 主机 名 。 执 行 
结果 如 图 16-11 所 示 。 























代码 清单 16-10 ”client.php 





<?php 

print "你 的 IP 地 址 是 :"; 
print getenv ("REMOTE ADDR"); 

Drint. Wehbe tT; 

print " 你 的 主机 名 是 :"; 

print gethostbyaddr (getenv ("REMOTE ADDR" ) ) ; 


起 
PEL MDESY; 
1 





print " 你 的 浏览 器 是 :" 7; 


print getenv ("HTTP USER AGENT"); 


BILt: Vebys:. Ty 














人 














(< 守 )(D | 回 hitp://localhost/clientph} PD ~ 上 | 园 ecanos X | 他 挛 计 量 


你 的 IP 地 址 是 :127.0.0.1 

你 的 主机 名 是 :DESKTOP-8AAOODU 

你 的 浏览 器 是 :Mozilla/5.0 (Windows NT 10.0: WOW64: Trident/7.0: rv:11.0) like 
Gecko 

















图 16-11 执行 结果 
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16.5 ”循环 处 理 


16.4 ”比较 运算 符 


后 面 介 绍 的 循环 处 理 和 条 件 判 断 会 涉及 判断 是 否 符合 设置 条 件 的 内 容 。 这 时 就 需要 用 到 比较 运 
算 符 。PHP 中 可 以 使 用 的 比较 运算 符 如 表 16-5 所 示 。 
































表 16-5 比较 运算 符 


比较 运算 符 内 容 



























































a==b a 等 于 b 
a>b 主 大 于 地 
a>=b 去 枯 手 等 手 卫 
a<b a 小 于 b 
a<=b a 小 于 等 于 b 
a<>b a 不 等 于 b 
如 果 符 合 条件 ， 则 为 “TRUE” 或 “ 真 ”; 如 果 不 符合 条 件 ， 则 为 “FALSE” 或 “ 假 ”。 需 要 注 








意 的 是 ， 表 示 相 等 的 “==” 需 要 连续 写 两 个 “=”。 
比如 ,“2 > 1” 是 正确 的 ， 所 以 是 “TRUE”( 真 )。 























16.5 ”循环 处 理 


用 程序 执行 的 优点 之 一 是 可 以 反复 执行 相同 的 处 理 。 下 面 就 来 介绍 一 下 执行 循环 处 理 的 for 
和 while 的 语法 。 











@16.5.1 通过 for 实 现 循环 


P 什么 是 for 

for 会 准备 一 个 计数 器 变量 ， 然 后 不 断 增加 该 计数 器 变量 的 值 ， 同 时 循环 执行 指定 范围 内 的 处 
理 。for 中 会 事先 设置 好 “ 当 计 数 右 变量 小 于 等 于 10 时 进行 循环 ”这 样 的 条 件 ， 如 果 不 满足 这 个 
条 件 ( 即 条 件 的 结果 为 FALSE )， 就 会 结束 循环 。 

for 的 语法 如 下 所 示 。 
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for 


;for ( 初始 值 ; 循环 条 件 ; 增 量 ) { 
循环 岳 行 的 处 理 





循环 执行 用 “{ }” 括 起 来 的 处 理 。 这 部 分 处 理 可 以 使 用 多 行 来 描述 。for 后 面 的 () 里 设置 了 
如 何 进行 循环 处 理 。 




















P 使 用 for 


我 们 很 难 用 语言 来 描述 如 何 使 用 Eor， 还 是 实际 操作 一 下 比较 好 。 试 着 输入 代码 清单 16-11 中 
的 脚本 。 代 码 清单 16-11 是 将 字符 “类 ”反复 显示 15 次 的 示例 。 执 行 结 果 如 图 16-12 所 示 。 


代码 清单 16-11 for.php 











<?php 

fOrl(Si= 1 Se= T1900 SIS 
print 火 "5; 

} 

| 

















乌有 | 回 http://localhost/forphp Pro | 国 localhost x | | 介 区 远 @ 


炒米 炒米 炒米 米 米 炒米 炒米 炒米 米 











图 16-12 执行 结果 
如 果 脚 本 没有 顺利 运行 ， 就 需要 一 个 字符 一 个 字符 地 进行 检查 了 。 看 看 “;”“{” 等 符号 是 否 
出 现 了 错误 。 




















PP” for 的 执行 流程 


下 面 我 们 来 看 一 下 代码 清单 16-11 的 内 容 。 在 国 这 部 分 代码 中 ，foz 后 面 的 () 里 的 内 容 有 如 
下 含义 。 

















2 $i=1 一 变量 $i 的 初始 值 是 1 ( 初始 值 ) 
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SS si<=15 ”一 当 $i 小 于 等 于 15 时 进行 循环 ( 循环 条 件 ) 
S $i=$i+1 一 每 次 循环 都 加 1 ( 增 量 ) 





具体 的 执行 流程 如 图 16-13 所 示 。 
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+ ' em | 
+ ' [em | 
+ 1 [em | 
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图 16-13 ”for 的 执行 流程 
在 这 个 示例 中 ，si 是 作为 计数 器 变量 来 使 用 的 。 计 数 器 变量 可 以 是 $1oop， 也 可 以 是 $p 或 
者 其 他 内 容 。 但 不 知 为 何 很 早 之 前 就 决定 了 使 用 $i 来 表示 计数 器 变量 。 














计数 器 变量 $i 小 于 等 于 15 的 情况 下 会 循环 执行 该 处 理 。 





初始 值 部 分 为 Si=1， 它 表示 计数 器 变量 的 第 一 个 值 是 1。 循 环 条 件 为 $i<=15。 也 就 是 说 ,在 


增 量 部 分 为 $i=$i+1。 它 表示 $i 每 次 增加 1。 在 程序 的 世界 里 ,“=” 表 示 将 右边 的 值 
( $i+1 ) 赋 给 左边 的 变量 ( $i )。 也 就 是 给 $i 加 1， 再 将 加 1 后 的 结果 赋 给 $i。 

由 于 $i=1， 所 以 初始 值 是 1。 在 这 个 基础 上 通过 $i+1 给 $i 加 1， 值 变 为 2， 然后 把 2 赋 给 
左边 的 $i， 于 是 $i 变 为 2。 接 着 再 通过 $i+1 给 $i 加 1， 值 变 为 3， 然 后 将 3 赋 给 $i，$i 就 变 























为 3…… 以 此 类 推 。 








$=$i+1 可 以 简写 为 $i++， 表 示 递 增 ( increment )。 也 就 是 说 ， 即 使 将 代码 清单 16-11 的 示 
例 按照 如 下 方式 进行 编写 ,结果 也 不 会 发 生 任何 改变 。$i=1 表示 从 1 开始 ，$i<=15 表示 小 于 等 


于 15 的 条 件 ，$i++ 表示 给 $i 加 1 ( 国 )。 
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代码 清单 16-12 for_variation.php 





<?php 

for ($i=1;$i<=15;$i++){ 
es 二 光 光 

} 

人 











S 16.5.2 ”通过 while 实现 循环 


下 面 将 介绍 如 何 通过 while 来 实现 循环 。 这 次 在 符合 循环 条 件 (TRUE ) 的 情况 下 循环 执行 处 
理 ， 具 体 语 法 如 下 所 示 。 


while 




















0 

昌 

S 
> 
UE 
i 

六 





1 

while 中 没有 设置 计数 带 变 量 初始 值 的 部 分 ， 所 以 我 们 必须 自己 设置 初始 值 。 而 且 在 循环 过 程 
中 一 定 要 有 不 符合 ( FALSE ) 循环 条 件 的 时 候 。 如 果 始 终 符 合 创建 的 循环 条 件 ， 处 理 就 会 无 限 循环 
下 走 。 





P 使 用 while 


前 面 用 for 写 入 了 15 次 “x*”， 这 次 我 们 使 用 while 来 实现 相同 的 内 容 (图 16-14 )。 

代码 清单 16-13 中 首先 准备 了 计数 器 变量 $i， 并 将 1 赋 给 了 这 个 变量 。 然 后 ， 每 执行 一 次 
“print " 火 ";” 就 给 $i 加 1， 并 用 while 循环 这 一 过 程 。 循 环 的 条 件 是 “变量 $i 小 于 等 于 
I 





























代码 清单 16-13 ”while.php 





<?php 
Gl 
while ($i<=15){ 
a 
$i++; 
} 














“$i=1;” 将 1 赋 给 了 $i ( 国 )， 之 后 判断 条 件 。 因 为 $i 的 初始 值 是 1， 所 以 符合 条 件 <=15 
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16.5 ”循环 处 理 


(小 于 等 于 15 )， 于 是 执行 print " x "， 然 后 “类 ”就 会 显示 出 来 ( 回 )。 

接 下 来 “$i++;” 给 $i 加 1 (一 前 一 节 )， 于 是 值 变 为 2 (图 )。 然 后 再 次 回 到 while 部 分 ， 
判断 值 是 否 符合 条 件 while ($i<=15 )。 

循环 这 一 过 程 ， 最 后 通过 “$i++;” 将 $i 增加 到 16， 这样 就 不 符合 (FALSE ) 设置 的 条 件 
了 ， 所 以 第 16 次 循环 结束 时 不 会 输出 “x*”。 

















$i=1 判断 是 否 小 于 等 于 15 














cs 输出 “x 
print "x* 
i 给 $i 的 值 加 1 
| $it+; 0 
$i-15 
$i=16* 一 当 $i 的 值 为 16 时 结束 循环 











16-14 ”while 的 执行 流程 


16.5.3 ”通过 do...while 实现 循环 





日 


与 while 相似 的 语法 还 有 do...while。 与 while 不 同 的 是 ，do...while 判断 条 件 的 操作 是 
在 循环 语句 之 后 进行 的 。 
do...while 的 语法 





循环 的 处 理 
; }while( 循环 条 件 ) 


执行 代码 清单 16-14 中 的 代码 ， 将 会 得 到 和 代码 清单 16-13 一 样 的 结 


代码 清单 16-14 do_while.php 





<?php 


$i=1; 
do{ 

Print wR Wy 

$i++; 
}while ($i<=15) 


?> 











通过 $i=1 将 初始 值 设置 为 1 ( 国 )， 然 后 通过 $i++ 给 $$ 加 1 ( 回 )， 最 后 通过 while ($i<=15) 
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判断 $i 的 值 是 否 小 于 等 于 15 (图 )， 这 就 是 整个 处 理 过程 。 

需要 注意 的 是 ,在 使 用 while 的 情况 下 会 先进 行 条 件 判断 ， 如 果 不 满足 条 件 ， 就 不 会 执行 循 
环 语句 ， 直 接 结束 。 但 在 dao...while 的 情况 下 ， 条 件 判断 会 放 到 最 后 ， 所 以 至 少 会 执行 一 次 循环 
语 人 名。 








16.6 条件 判断 


if 是 让 执行 的 内 容 随 着 条 件 的 变化 而 变化 的 处 理 。 在 PHP 脚本 中 ，if 的 语法 如 下 所 示 。 
if 的 语法 





; if (条件 ){ 
符合 条 件 时 执行 的 处 理 
: elsel{ 


不 符合 条 件 时 执行 的 处 理 














如 果 符 合 () 内 的 条 件 (TRUE)， 则 执行 “符合 条 件 时 执行 的 处 理 ”， 如 果 不 符合 条 件 
( FALSE )， 则 执行 “不 符合 条 件 时 执行 的 处 理 ”。 
P if 的 执行 流程 

试 着 执行 代码 清单 16-15 中 的 代码 。 执 行 结果 如 图 16-15 所 示 。 


代码 清单 16-15 ”condition.php 





<?php 
if (200>100){ 
Toe nae I Se 
}elsef{ 
BE 


国 四 国 图 


} 


?> 
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/ http://localho.../condition.php x 





所 [8) ni 


大 








图 16-15 ”执行 结果 
a>b 表示 a 大 于 b (一 16.4 节 )。 如 果 问 200 和 100 哪个 大 ， 当 然 是 200。 也 就 是 说 ， 辆 用 于 
判断 “200 大 于 100”( 200 > 100 ) 是 否 正确 。 因 为 这 是 正确 的 ， 所 以 会 执行 田中 的 “符合 条 件 时 执 
行 的 处 理 ”。 
如 果 不 符合 国 的 条 件 〈 圆 )， 则 执行 圆 。 回 中 的 处 理 就 不 会 再 执行 了 。 
图 16-16 显示 了 if 的 执行 流程 。 





























( ee = 一 | 符合 条 件 时 执行 的 处 理 


} else | 


(smo | 


| 


16-16 if 的 执行 流程 
当然 在 实际 的 脚本 中 ， 不 会 只 设置 “200 > 100” 这 种 简单 的 条 件 。if 也 可 以 用 在 比较 重要 的 
处 理 上 ， 比 如 判断 是 否 连 接 到 了 数据 库 ， 或 者 判断 是 否 包 含 了 指定 的 字符 等 。 
另外 ,关于 else | 不 符合 条 件 时 ~ | 的 部 分 ， 如 果 不 需要 的 话 可 以 省 略 。 在 这 种 情况 下 ， 只 
有 当 符合 条 件 时 才 会 执行 指定 的 处 理 ， 如 果 不 符合 条 件 ， 则 不 执行 任何 操作 。 
































16.6.2 ”三 元 运算 符 


如 果 只 是 根据 条 件 来 修改 值 ， 语 法 其 实 非常 简单 。 下 面 我 们 来 看 一 下 三 元 运算 符 。 
三 元 运算 符 





ooosoooooooooooossssoeooo eeeossasseosoo eos oseooo ooo esose oooorooo oo oss oo ooo eos eosoo ooo ooo esos ooo ooo eosos ees e eosssosse 上 ose 
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三 元 运算 符 简 化 了 if 的 编写 ， 并 根据 条 件 改 变 了 值 本 身 。 注 意 输入 正确 的 “?” 和 “:” 等 符 
号 。 下 面 是 通过 三 元 运算 符 来 表示 前 一 他 介绍 的 “如 果 200 > 100， 则 显示 " 大 "， 否 则 显示 "小 "” 
的 示例 。 


代码 清单 16-16 ternary.php 





























<?php 
print (200>100)?" 大":" 小“ 


?> 





























下 面 是 使 用 三 元 运算 符 实 现 “ 上 午 12 点 之 前 显示 "上午"， 否 则 显示 "下 午 "” 的 示例 。 
代码 清单 16-17 am_pm.php 





<?php 
print (date("G")<12)?" 上 午 ":" 下 午 "; 


到 














16.6.3 ”设置 了 多 个 条 件 的 if 的 语法 


@ 一 一 全 上 上 上 上 上 
if 可 以 设置 多 个 条 件 表达 式 ， 有 具体 语法 如 下 所 示 。 
设置 了 多 个 条 件 的 if 的 语法 


: if 条 件 1){ 

: 符合 条 件 1 时 执行 的 处 理 
: }elseif (条 件 2)f{ 
: 符合 条 件 2 时 执行 的 处 理 
: }elseif (条 件 3){ 
符合 条 件 3 时 执行 的 处 理 











else{ 


不 符合 所 有 条 件 时 执行 的 处 理 





设置 了 多 个 条 件 的 if 会 循环 执行 以 下 处 理 。 





如 果 符 合 “条 件 1”( TRUE )， 则 执行 “符合 条 件 1 时 执行 的 处 理 ”， 如 果 不 符合 ， 则 转 到 下 一 


D4 
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如 果 符 合 “ 条 件 2”( TRUE )， 则 执行 “符合 条 件 2 时 执行 的 处 理 ”， 如 果 不 符合 ， 则 转 到 下 一 
当 所 有 条 件 都 不 符合 时 ， 执 行 “ 不 符合 所 有 条 件 时 执行 的 处 理 ”。 


存储 过 程 中 的 条 件 分 支 
虽然 本 书 没有 涉及 ， 但 是 在 MySOQL 存储 过 程 中 也 可 以 使 用 if 进行 条 件 判 断 。 























PP 设置 了 多 个 条 件 的 if 语句 的 执行 流程 

下 面 来 看 一 个 复杂 一 些 的 处 理 。 我 们 试 着 让 问候 语 随 着 访问 时 间 发 生变 化 。 当 前 时 间 可 以 通过 
date("G") 获取 (一 16.3.2 节 ) 也 就 是 说 ,需要 设置 “date ("G") 的 值 大 于 等 于 18” 或 者 
“date ("G") 的 值 大 于 等 于 9” 这 样 的 条 件 ， 然 后 通过 if 进行 分 支 处 理 。 
具体 来 说 ， 就 是 创建 代码 清单 16-18,， 在 0 点 到 6 点 显示 “不 困 吗 ”,， 在 6 点 到 9 点 显示 “早上 
好 ”, 在 9 点 到 18 点 显示 “你 好 ”, 在 18 点 到 0 点 显示 “晚上 好 ”。 执 行 结果 如 图 16-17 所 示 。 


代码 清单 16-18 if.php 
































<?php 

if (date("G")>=18){ 
print "晚上 好 "，; 

}elseif (date("G") >=9){ 
print "你 好 "; 

}elseif (date("G") >=6){ 
print "早上 好 "; 

}else{ 
print "不 困 吗 "; 

















} 


闫 沟 




















-) 园 http://localhostitphp D7 | Iocalhost x 
早上 好 














16-17 执行 结果 
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可 以 看 到 ， 根 据 当 前 的 具体 时 间 ， 浏 览 器 上 显示 了 相应 的 字符 串 。 


16.6.4 ”使 用 了 switch 的 条 件 判断 


当 根 据 变 量 值 执行 不 同 的 处 理 时 ， 我 们 可 以 使 用 switch ( 见 图 16-18 )。 例 如 ， 当 变量 $i 为 
1 时 执行 x x ; 当 变量 $i 为 2 时 执行 AA; 当 变 量 $i 为 3 时 执行 OO。 
使 用 前 一 节 的 i£ 也 可 以 执行 相同 的 处 理 ， 但 使 用 switch 编写 更 容易 让 人 理解 。 
switch 的 语法 如 下 所 示 。 
switch 的 语法 


























: switch( 变量 ) { 
; case 变量 的 值 1 : 
处 理 1 

; break; 

; case 变量 的 值 2: 
: 处 理 2 





: default: 
不 符合 所 有 条 件 时 执行 的 处 理 





如 果 switch 之 后 设置 的 变量 和 case 之 后 描述 的 值 相符 ， 则 执行 相应 的 处 理 。 请 务必 在 各 处 
理 的 最 后 加 上 break。break 是 用 于 结束 处 理 的 命令 。 
如 果 不 符合 所 有 条 件 ， 则 执行 “default :” 后 面 编写 的 处 理 。 


根据 它 来 判断 


switch ( 变量 ) | 














case O 〇 : 


当 变 量 值 为 OO 时 方 
case 人 人 : 在 要 后 吕 | 
当 变 量 值 为 AA 时 > 


break; 
default ， 


当 不 符合 所 有 条 件 时 = > 
| 


16-18 ”switch 的 执行 流程 
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P Switch 语法 的 执行 流程 
例如 ， 代 码 清 单 16-19 是 在 10 点 的 时 候 显 示 “10 点 的 零食 ”， 在 15 点 的 时 候 显示 “3 点 的 零 
食 ”, 在 10 点 和 15 点 之 外 的 时 间 显 示 “ 这 不 是 零食 ”的 示例 。 执 行 结果 如 图 16-19 所 示 。 


代码 清单 16-19 snack.php 





<?php 
switch(date("G")){ 
case 10: 
print "10 点 的 零食 " ; 
break; 





case 15: 
print "3 点 的 零食 " ; 





break; 
default: 

print " 这 不 是 零食 " ; 
} 


Es 





























这 里 再 强调 一 遍 ， 不 要 忘记 在 各 处 理 的 后 面 加 上 break。 如 果 没 有 写 break， 后 面 的 处 理 也 
会 被 执行 。 男 外 ， 注 意 不 要 写 错 “case 变量 的 值 1:” 中 的 “:”。 

















http://localhost/snack.php x 


所 | 加 localhost/snackphp > CC& | 区 Search 





这 不 是 零食 








16-19 ”执行 结果 
下 面 我 们 来 思考 一 下 如 何 使 用 switch 创建 如 下 所 示 的 图 案 。 


OOVAO 太 OVAO 妈 OVAO 六 OVAO 太 OVAO 妈 OVAO 友 OVA…… 





具体 方法 是 使 用 for (一 16.4 节 ) 将 1 ~ 5 赋 给 变量 , 使 用 switch 分别 对 每 个 值 进行 不 同 
的 处 理 。 然 后 使 用 for 循环 这 一 过 程 。 

变量 $x 的 值 为 1 则 输出 ，2 则 输出 支 ，3 则 输出 O，4 则 输出 Y，5 则 输出 A， 这 样 重复 循环 8 
次 ， 就 可 以 连续 输出 8 次 “OO 支 O 〇 YA” 的 图 案 。 具 体 请 看 代码 清单 16-20。 执 行 结果 如 图 16-20 所 示 。 
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代码 清单 16-20 switch.php 





<?php 
for ($x=1; Sx<=8; Sx++) { 


回 


for (Sy=1; Sy<=5;$y++) { 
switch($y){ 
case 1:; 


国 国 


print ™ OO ™; 
break; 

Case 2: 

Bt 
break; 

CAaser 3 
Drint TO 
break; 

Case 4: 

第 下 IE 
break; 

Case 5: 
print "AA"; 
break; 














一 四 | x 
(© OBrien PD OG ccamon x 加 | 价 六 次 国 
加 太 CV 入 加 六 OV 和 加 六 OV 和 加 六 CV 入 加 六 OV 和 妈 加 文生 回 太 oY 和 加 六 OV 入 




















图 16-20 ”执行 结果 


我 们 来 看 一 下 处 理 的 内 容 。 
首先 按照 “变量 $y 是 1 则 写 出 QO， 变量 $y 是 2 则 写 出 太 ” 的 方式 进行 分 支 处理 〈 国 )。 
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然后 将 switch 语句 以 for 的 形式 进行 循环 ( 回 )。 





变量 $y 的 值 会 按照 1、2、3、4 、5 的 方式 递增 ， 因 此 会 分 别 和 输出 “@”“ 女 ”“O”“V7 “全 ”。 
将 这 个 使 用 了 计数 器 变量 $y 的 £or 语句 ， 再 通过 使 用 了 计数 器 变量 $x 的 for 语句 循环 8 
次 (图 )。 





注意 ,“{” 和 “} ”必须 成 对 出 现 。 


全 16.7.1 什么 是 数组 


前 面 使 用 的 变量 都 和 $i=1000 一 样 只 存储 了 一 个 值 。 但 其 实 还 有 可 以 存储 多 个 值 的 特殊 变量 。 
这 种 特殊 变量 称 为 数组 。 

声明 某 变量 不 是 普通 变量 而 是 数组 变量 的 方法 有 很 多 种 ， 我 们 先 来 介绍 一 种 使 用 了 arzay 命 
令 的 方法 。 数 组 变量 的 命名 规则 和 16.1.2 节 介 绍 的 规则 相同 。 
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例如 ， 将 变量 $m 定义 为 数组 ， 然 后 通过 下 面 的 方法 使 其 存储 大 量 数据 。 





$m=array ( "老鼠 " nm 和牛" "老虎 " "兔子 " , "Dragon"); 








这 样 ， 变 量 $m 中 就 存储 了 “老鼠 ”"”“ 牛 "”“ 老 虎 ”“ 兔 子 ”“Dragon” 这 5 个 字符 串 。 一 旦 赋 了 
值 ，PHP 就 会 自动 设置 数据 类 型 (一 16.1.4 节 )， 因 此 $m 是 字符 串 类 型 的 数组 变量 。 

下 面 是 从 数组 中 取出 各 值 的 方法 。 例 如 ， 上 面 示 例 中 的 值 (字符 串 ) 从 左 到 右 分 别 以 $m [0] 、 
$m[1] 、$m[2] 、$m[3] 、$m[4] 的 名 称 保存 了 起 来 。 我 们 可 以 通过 指定 名 称 来 取出 相应 的 值 。 也 
就 是 说 ， 执行 “print Sm[2] ; ”就 会 输出 “老虎 "。 注 意 输出 的 并 不 是 “ 牛 ”。 保 存 的 值 和 相应 的 
数组 变量 如 表 16-6 所 示 。 





也 





























表 16-6 保存 的 值 和 相应 的 数组 变量 



































数组 变量 保存 的 值 
$m[0] 老鼠 
$m[1] 午 
$m[2] 老虎 
$m[3] 兔子 
$m[4] Dragon 
我 们 可 以 通过 [] 中 的 数字 判断 这 是 第 几 号 元 素 的 值 。 写 在 [] 中 的 数字 编号 称 为 下 标 。 下 标 

















是 从 0 开始 的 。 大 家 可 能 觉得 下 标 从 0 开始 会 有 些 奇怪 ,但 是 在 计算 机 领域 ,数字 通常 都 是 以 0 开 
头 的 。 因 此 ， 数 组 最 后 一 个 下 标的 值 要 比 数组 元 素 的 个 数 小 1， 这 一 点 需要 注意 。 
例如 ， 在 上 面 的 示例 中 ， 第 5 个 元 素 “Dragon” 是 sm[4] ， 而 不 是 Sm[5] 。 














在 不 使 用 array 命令 的 情况 下 ， 给 数组 赋值 的 方法 有 以 下 几 种 。 














P 直接 赋值 


$m[4] ="Dragon"; 
$m[0] =" 老 鼠 "7 
TIE 
$m[1]=" 牛 "; 
$m[3] =" 免 于 "; 








在 这 种 情况 下 ， 无 论 下 标 顺 序 如 何 ， 都 可 以 进行 赋值 。 即 使 像 Sm [777] = "学问" 这 样 使 用 一 
些 奇 怪 的 下 标 也 没有 关系 。 
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PP 按 顺序 赋值 


Sm[] =" 老 鼠 "; 
$m[] =" 牛 "， 

$m[] =" 老 虎 "; 
$m[] =" 兔 子 "; 


sm[] ="Dragon"; 








使 用 该 方法 会 按照 sm[0] 、$sm[1] 、$m[2] 、$m[3] 、$m[4] 的 顺序 进行 赋值 。 当 使 用 for、 
while 执行 循环 处 理 时 ， 该 方法 会 非常 方便 。 
在 没有 熟悉 下 标的 使 用 方法 之 前 很 容易 出 错 。 所 以 大 家 在 使 用 下 标的 时 候 要 多 加 注意 。 

















”使 用 数组 的 示例 


代码 清单 16-21 是 利用 数组 对 代码 清单 16-18 进行 改良 的 示例 。 将 “不 困 吗 ”“ 早 上 好 ”“ 你 
好 ”“ 晚 上 好 ” 赋 给 数组 变量 $m， 如 果 当 前 时 间 在 0 点 到 6 点 ， 则 显示 “不 困 吗 ”在 6 点 到 9 点 
则 显示 “早上 好 ”， 在 9 点 到 18 点 则 显示 “你 好 ”, 在 18 点 到 0 点 则 显示 “晚上 好 ”。 


代码 清单 16-21 array.php 


















































<?php 
$m=array (" 不 困 吗 "," 早上 好 "," 你 好 "," 晚上 好 "); 
if (date("G")>=18){ 
print Sm[l3]3 
}elseif (date("G") >=9){ 
print Sml[l2]: 
}elseif (date("G") >=6){ 
peint SmLllls 
}else{ 
print sm[0]; 























} 


?> 











贺 中 的 “不 困 吗 ” 赋 给 了 sm [0] ,“ 早 上 好 ” 赋 给 了 $m [1] ,“ 你 好 ” 赋 给 了 sm [2] ,“ 晚 上 
好 ” 赋 给 了 $m[3]。 


下 标的 数值 设 定 要 恰当 
明明 只 使 用 了 几 个 下 标 ， 却 要 像 sm[777] =" 学 问 " 这 样 设置 一 个 奇怪 的 数值 ， 这 种 做 法 会 造成 内 存 的 浪费 1。 
最 好 根据 需要 来 设置 下 标 值 。 









































四 本 节 介绍 的 是 以 数字 ID 作为 键 名 的 数组 ， 即 索引 数组 (associative array )， 数 组 中 的 各 元 素 在 内 存 中 按照 先后 顺 
序 存 储 在 连续 的 内 存 块 中 并 通过 索引 访问 ， 所 以 下 标 过 大 会 导致 预 分 配 的 内 存 较 大 。 译 者 注 
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全 16.7.3 关联 数组 

对 于 下 标 ， 除 了 可 以 使 用 表示 顺序 的 数字 之 外 ， 还 可 以 使 用 用 户 自 定义 的 字符 串 。 这 种 数组 称 
为 关联 数组 ( associative array )。 

例如 ， 我 们 可 以 使 用 保存 了 考试 分 数 的 数组 变量 来 设置 直观 易 懂 的 下 标 ， 例 如 英语 分 数 可 以 设 
置 成 St ["English"] ， 语 文 分 数 可 以 设置 成 St ["Chinese"] 。 

代码 清单 16-22 是 使 用 关联 数组 的 示例 。 执 行 结果 中 会 显示 “总 分 是 : 225”。 


代码 清单 16-22 ”associative.php 

















<?php 

st[l"English"]=s73; 

st["Math"]=84; 

st["Chinese"] =68; 

Brint " 总 分 是 :"; 

print s$t[l"English"]+$t["Math"] +$t["Chinese"]; 


> 








关联 数组 的 下 标 即 使 不 使 用 "" 或 “ 括 起 来 也 能 正常 工作 吗 
从 规范 的 角度 来 说 ， 关 联 数 组 的 下 标 需 要 用 "" 或 '' 括 起 来 。 但 是 ， 即 使 关联 数组 的 下 标 像 
St [English] 这 样 没有 用 "" 或 '' 括 起 来 ， 很 多 时 候 代码 也 能 正常 运行 。 















































戎 不 规范 示例 ( associative_no.php ) 


<?php 
sm[Chemistry] =100; 
sm[Physical] =200; 


print $m[Chemistry]; 


oh sa yoo I oF oo 
print $m[Physicall] 


?> 























执行 上 面 的 代码 通常 会 正常 显示 100 和 200。 但 是 严格 来 说 这 种 写法 是 错误 的 ， 正 确 的 写法 应 该 是 
sm["Chemistry"] 和 $m["Physical"]。 昌 然 写 成 $m[Chemistry] 和 $m[Physical] 不 会 报错 ， 但 是 
实际 上 内 部 已 经 发 生 了 错误 。 之 所 以 没有 报错 ， 是 因为 PHP 尽 了 最 大 的 努力 去 编译 。 

不 管 怎样 ， 关 联 数组 的 下 标 最 好 要 用 "" 或 '' 括 起 来 。 



































16.8 总 结 


本 章 介绍 了 以 下 内 容 。 


@ PHP 中 变量 的 使 用 方法 

@ PHP 中 字符 串 的 处 理 方法 

@ PHP 函数 的 概况 和 一 些 常用 函数 
@ 比 较 运 算 符 的 使 用 方法 

@ 循 环 处 理 的 程序 设计 

@ 条 件 判断 的 程序 设计 

@ 数 组 的 使 用 方法 


本 章 介绍 的 内 容 都 是 今后 和 MySQL 一 起 使 用 时 必须 了 解 的 内 容 ， 对 初次 接触 编程 语言 的 人 来 


























说 格外 重要 。for 、while 和 switch 等 都 是 编程 的 基础 内 容 ， 请 大 家 熟练 掌握 相关 的 使 用 方法 。 





> 自我 检查 
下 面 检查 一 下 本 章 学 习 的 内 容 是 否 全 部 理解 并 掌握 了 。 





口 能 够 理解 PHP 变量 的 命名 规则 

口 能 够 理解 PHP 中 连接 字符 串 的 方法 。 

口 能 够 理解 用 " " 将 变量 括 起 来 和 用 ，“' 将 变量 括 起 来 有 什么 不 同 
口 能 够 显示 日 期 和 时 间 

口 能 够 显示 客户 端的 IP 地 址 和 Web 服务 器 的 信息 

口 了 解 如 何 使 用 foz 进行 循环 处 理 

口 了 解 如 何 使 用 while 进行 循环 处 理 

口 了 解 如 何 使 用 if 进行 条 件 判断 

口 了 解 如 何 使 用 switch 进行 条 件 判 断 

口 了 解数 组 的 使 用 方法 





PHP 的 文档 


本 书 以 介绍 MySQL 为 主 ， 所 以 只 介绍 了 PHP 最 基本 的 知识 。 下 面 的 网 站 中 有 PHP 的 中 文 文档 ( 见 图 16-21 )， 
如 果 有 不 明白 的 地 方 可 以 自行 查找 。 与 MySQL 不 同 ， 该 文档 会 不 断 进行 更 新 。 


PHP 的 中 文 文档 
@ http://php.net/manual/zh/index.php 
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(€) OphpneumanualzManguageoperatorscomparsonphp 


Downlaads Documentation ICeilnvofvedlela 


Change language: Ginesxe Ginplifed ~ 
Edit «Reporta Bug 


比较 运算 符 


比较 运算 符 ， 如 同 它们 各 称 所 暗示 的 ， 罗 许 对 两 个 值 进行 比较 。 还 可 以 参考 PHP 类 型 比 
较 雪 看 不 同类 型 相互 比较 的 例子 。 

比较 运算 符 
例子 名称 结果 


$a= 等 + TRUE ,如果 类 漠 转 换 后 90 等 十 Sb- 
Ei 


$a=== 全 等 TRUE , 如 果 $a 等 于 $b， 并 且 它 们 的 类 型 也 相同 。 








16-21 PHP 的 中 文 文档 


te 练习 题 





问题 1 





























strtotime 是 将 日 期 字符 串 解析 为 时 间 戳 ( 可 参考 16.3.2 节 和 的 提示 ) 的 函数 。 使 用 strtotime 
函数 和 date 函数 能 够 以 时 间 格 式 获 取 从 今天 开始 到 某 天 后 的 所 有 日 期 。 请 使 用 该 函数 ， 像 图 16-22 一 样 
以 “年 月 日 ( 英语 的 星期 ”的 格式 显示 从 今天 开始 到 30 天 后 这 段 时 间 的 所 有 日 期 。 
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以 时 间 格 式 获 取 从 今天 开始 到 某 天 后 这 段 时 间 的 所 


计 
黎 











园 htpv/localhostycalendar| DO ] | 国 ,ocanost x 介 交 半 @ 


2018/03/12 (Monday) 
2018/03/13 (Tuesday) 
2018/03/14 (Wednesday) 
2018/03/15 (Thursday) 
2018/03/16 (Friday) 
2018/03/17 (Saturday) 
2018/03/18 (Sunday) 
2018/03/19 (Monday) 
2018/03/20 (Tuesday) 生 
2018/03/21 (Wednesday) 义 今天 是 2018 年 3 
2018/03/22 (Thursday) 省 一 一 yy 
2018/03/23 (Friday) 12 日 为 例 
2018/03/24 (Saturday) 
2018/03/25 (Sunday) 
2018/03/26 (Monday) 
2018/03/27 (Tuesday) 









































2018/03/28 (Wednesday) 

2018/03/29 (Thursday) 

2018/03/30 (Friday) 

2018/03/31 (Saturday) 

2018/04/01 (Sunday) 

2018/04/02 (Monday) 

2018/04/03 (Tuesday) 

2018/04/04 (Wednesday) 四 





16-22 ”以 时 间 格 式 获取 从 今天 开始 到 某 天 后 这 段 时 间 的 所 有 日 期 












































[HINT | PHP 中 日 期 和 时 间 相关 的 函数 使 用 了 UNIX 时 间 戳 。 我 们 可 以 通过 在 date 函数 ( 一 16.3.2 节 ) 
| ef 」 的 第 2 个 参数 中 指定 时 间 蕉 来 返回 日 期 和 时 间 。 时 间 稚 可 以 应 用 于 各 个 方面 ， 如 万 年 历 等 。 









































口 
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i 参考 答案 








问题 1 











创建 代码 清单 16-23 中 的 PHP 脚本 。 


代码 清单 16-23 示例 calendar.php 





<?php 

for ($i=0;$i<=30;$i++){ 
print date("Y/m/d (1)",strtotime("now +$i day")); 
beint vebEy>y 
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HTML 是 使 用 PHP 运行 MySQL 的 基础 。 本 章 我 们 将 学 

























































































习 HTML 的 相关 知识 以 及 通过 Web 页 面 发 送 数据 和 接收 数据 的 
方法 。 

当 使 用 MySQL+PHP 这 样 的 Web 应 用 程序 时 ， 这 些 知 识 是 
不 可 或 缺 的 。 


17.1 HTML 源 代码 





我 们 每 天 都 会 在 浏览 器 上 浏览 Web 页 首 











ji。 你 知道 Web 页 面 的 内 部 是 什么 样子 的 吗 ? 























从 本 章 开始 到 最 后 我 们 都 会 制作 Web 页 下 








i。 先 来 确认 一 下 从 Web 服务 融 发 来 的 Web 页 面 的 内 容 。 























我 们 要 让 源 代 码 显示 在 浏览 需 上 〈 见 图 
览 絮 , 在 页 了 




















控制 台 





17-1 )。 对 于 Chrome 、Internet Explorer 或 FireFox 等 浏 


i 上 单 击 右键 , 从 显示 的 上 下 文 菜单 中 选择 “查看 网 页 源 代 码 ” 即 可 让 源 代码 ”显示 出 来 。 


[ES 





- 母 




















yudFsjvr8DmczNaEtVDAMVVWKoAYhOkQl1_7 


4695sB_bTSptWvx3Q0dRX8eQasdOguk7ejw] 


二 

















script sr uring/js?v=29qhuba-JjWpkUZ6WyvMzCPWFr8_TrAdbOEp 
< 是 ETYwanyaN cio lnyveica% 
Cg 
图 17-1 Web 页 面 的 源 代码 





QD 在 Chrome、Internet Explorer 或 FireFox 等 六 


只 吗 


览 器 上 ， 也 可 以 通过 F12 按键 来 查看 源 代码 。 





译 者 注 
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显示 出 来 的 源 代码 其 实 是 一 个 HTML 文件 ， 这 个 文件 由 <html> 或 <a ~ /a> 等 用 “< >” 括 
起 来 的 字符 组 成 。 用 “< >” 括 起 来 的 对 象 称 为 标签 ( tag )， 表 示 在 这 里 显示 x x 内 容 、 在 这 里 显 
示 x x 的 图 片 文件 等 命令 。 

HTML 文件 是 按照 HTML 规则 制作 的 文本 文件 。 浏 览 妖 只 会 按照 从 Web 服务 器 发 来 的 HTML 
文件 的 命令 来 配置 一 起 发 来 的 文字 和 图 像 。 

















17.2 ”制作 Web 页 面 的 两 种 方法 


在 15.4 节 中 ， 我 们 讨论 了 基态 页 面 和 动态 页 面 的 相关 内 容 。 本 节 将 进一步 对 此 展开 讨论 。 
制作 Web 页 面 的 方法 有 以 下 两 种 。 









































@ 在 编辑 器 中 直接 输入 标签 ， 或 者 使 用 网 页 制作 软件 制作 标签 
@ 通 过 PHP 脚本 等 程序 编写 标签 


首先 来 看 第 1 种 制作 Web 页 面 的 方法 。 

HTML 文件 由 文本 组 成 。 只 要 理解 了 语法 规则 ， 就 可 以 在 记事 本 等 编辑 器 中 轻松 地 将 其 制作 出 
来 。 过 去 要 想 制 作 Web 页 面 ， 只 能 用 键盘 直接 输入 到 编辑 器 中 。 而 现在 即使 不 知道 标签 相关 的 知 
识 ， 只 要 使 用 Adobe Dreamweaver 、 网 页 制作 (HomePage Builder ) 等 网 页 制作 软件 ， 就 可 以 按照 
文字 处 理 软件 的 风格 在 制作 的 页 面 中 自由 地 生成 标签 。 
虽然 “在 编辑 器 中 直接 输入 标签 ”和 “使 用 软件 制作 标签 ”有 些许 不 同 ， 但 通过 第 1 种 方式 制 
作 的 Web 页 面 的 内 容 不 会 发 生 改 变 ，Web 页 面 也 只 会 显示 标签 指定 的 内 容 。 例 如 ， 对 于 一 个 内 容 
为 “晚上 好 ”的 Web 页 面 ， 任 何人 在 任何 时 候 访 问 该 页 面 ， 它 都 仅 会 显示 “晚上 好 ”。 这 样 的 Web 
页 面 称 为 静态 Web 页 面 。 
























































































































































下 面 来 看 第 2 种 制作 动态 Web 页 面 的 方法 。 
例如 ， 下 面 是 16.6.3 节 中 创建 的 PHP 脚本 。 











<?php 

if (date("G")>=18){ 
print "晚上 好 " 

}elseif (date("G")>=9){ 
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print "你 好 "; 
}elseif (date("G") >=6){ 

print "早上 好 "; 
}elsef{ 

print "不 困 吗 "; 
} 


?> 











执行 上 面 的 脚本 后 ， 浏 览 器 会 在 0 点 到 6 点 显示 “不 困 吗 ”在 6 点 到 9 点 显示 “早上 好 ”, 在 
9 点 到 18 点 显示 “你 好 ”, 在 18 点 到 0 点 显示 “晚上 好 ”。 之 所 以 会 出 现 这 样 的 情况 ， 是 因为 PHP 
根据 “时 间 ” 这 个 条 件 输 出 了 不 同 的 内 容 。 

像 这 种 通过 PHP 脚本 等 程序 间接 地 输出 显示 内 容 ， 而 且 显 示 内 容 会 根据 条 件 发 生 改 变 的 Web 
页 面 称 为 动态 Web 页 面 ( 见 图 17-2 )。 















































， 
上 





print" 晚 上 好 " 


图 17-2 动态 Web 页 面 


17.2.3 浏览 器 不 区 分 静态 页 面 和 动态 页 面 


第 1 种 方法 通过 编辑 器 或 网 页 制作 软件 来 输入 标签 ， 从 而 生成 HIML 文件 。 第 2 种 方法 通过 
PHP 等 程序 间接 地 、 动 态 地 制作 HTML 文件 。 对 浏览 器 而 言 ， 这 两 种 方法 并 没有 什么 区 别 。 因 为 
客户 端 显示 的 是 完全 相同 的 内 容 。 

我 们 的 目标 是 将 与 MySQL 交互 的 结果 制作 成 Web 页 面 并 将 其 发 送 回 客户 端 。 当 然 ， 交 互 的 结 
果 并 不 总 是 一 样 的 。 一 般 需 要 使 用 PHP 脚本 操作 MySQL， 然 后 通过 PHP 脚本 接收 MySQL 发 送 的 
结果 ， 动 态 制 作 Web 页 





















































o 





17.3 HTML 的 规则 








要 想 使 用 PHP 脚本 操作 MySQL， 就 需要 掌握 HTML 的 相关 知识 。 本 节 将 重点 介绍 这 部 分 内 
容 。 对 HIML 有 一 定 了 解 的 读者 可 以 直接 阅读 17.7 节 。 











17.3 HTML 的 规则 | 375 


HTML 有 各 种 各 样 的 版 本 。 本 书 主要 对 最 新 的 HTML 5?( 2017 年 7 月 的 最 新 版 本 ) 的 使 用 方法 
进行 介绍 。 
P HTML 文件 的 扩展 名 为 “.html”， 如 果 包 含 PHP 脚本 则 为 “.php” 


HTML 文件 是 由 <html > 等 标签 构成 的 文本 文件 。 扩 展 名 为 “.html” 或 者 “.htm”( 本 书 统一 
为 “html”)。 但 如 果 其 中 包含 了 PHP 脚本 ， 则 需要 以 “.php” 为 扩展 名 (一 15.8.1 节 )。 
































P 标签 的 写法 

HTML 通过 编写 <html > 之 类 的 标签 来 制作 Web 页 面 。 大 多 数 标 签 会 像 下 面 这 样 包含 “开始 
标签 ”和 “结束 标签 "， 加 上 二 者 之 间 夹 杂 的 字符 串 后 ， 整 个 部 分 称 为 元 素 。 下 面 是 <a> 标签 的 示 
例 ， 有 时 我 们 也 把 它 叫 作 a 元 素 。 


























属性 





<a _ href=" 链 接地 址 "> 链接 的 字符 串 </a> 






































始 标签 结束 标签 
元 素 





与 上 面 的 href 一 样 ,“ 开 始 标签 ”中 可 以 包含 一 些 具体 信息 ， 这 些 信 息 称 为 属性 。 

在 HTML 中 ,标签 和 属性 需要 用 半角 字符 编写 ， 字母 大 小 写 均 可 。 不 过 ,最近 比较 流行 使 用 
小 写字 母 ， 所 以 本 书 也 使 用 了 小 写字 母 。 

需要 注意 的 是 ,很 多 标签 不 需要 用 到 结束 标签 ， 比 如 后 面 介绍 的 <br> 标签 等 。 
































PP HTML 文件 的 结构 
HTML 文件 的 结构 如 下 所 示 。 





HTML 文件 的 结构 


; <!IDOCTYPE html> 
<html> 
7 <head> 
编写 页 面 标题 和 链接 的 关系 等 
: </head> 


: <body> 

编写 作为 Web 页 面 显示 的 主体 
: </body> 
: </html> 











Q 在 2018 年 9 月 翻译 本 书 的 时 候 ，HTML 的 最 新 版 本 为 HTML 5.2。 
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第 1 行 是 文档 类 型 声明 , 在 HTML 5 的 情况 下 ， 需 要 编写 为 <1DOCTYPE html >。 
之 后 使 用 <htm1> 和 </html> 将 整个 文档 括 起 来 ， 并 在 其 中 编写 head 元 素 和 body 元 素 。 
当然 也 可 以 编写 大 量 的 标签 ， 稍 后 会 对 此 进行 介绍 。 这 里 需要 记 住 页 面 上 显示 的 是 body 元 素 。 


PP HTML 文件 的 示例 


代码 清单 17-1 tag.html 






























































<!1DOCTYPE html> 
=iEmls 
<head> 
<title>SQL 咖啡 厅 的 页 面 </title> 

</head> 

<body> 
欢迎 光临 soz 咖啡 厅 ! 

</body> 

el 











在 HIML 5 中 ,文件 开头 要 编写 文档 类 型 声明 < !DoCTYPE html> ( 国 )。 

HTML 的 主体 部 分 以 <htmL> ( 回 ) 开始 ,以 </html> ( 回 ) 结 

<head> ~ </head> 中 编写 的 是 标题 和 链接 。<title> ~ </title> 中 编写 的 是 页 面 的 标 
题 。 这 个 标题 会 显示 在 浏览 器 的 工具 栏 上 ， 或 者 显示 在 书签 和 检索 结果 等 地 方 。 请 在 
<head> ~ </head> 中 编写 这 部 分 内 容 。 

<body> ~ </body> 是 主体 部 分 。 这 里 只 会 显示 文字 “欢迎 光临 Ser 咖啡 厅 !”。 
编写 完 程序 后 ， 将 其 保存 为 tag.html 并 存放 到 要 发 布 的 文件 夹 中 ， 这 时 不 要 忘记 将 字符 编码 设 
置 成 UTF-8。 操 作 完成 后 打开 下 面 的 URL ( 见 图 17-3 )。 


































































































http://localhost/tag.html 





| @ http://localhost/tag.html P - 和 | 碟 SQL 各 啡 厅 的 页 面 x 上 分 交 密 人 @ 
| 欢迎 光临 SQL 咖 啡 厅 ! 

| 

| 








17-3 ”执行 结果 
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> 用， 和， 将 属性 括 起 来 
假设 现在 我 们 要 设置 链接 http://www.ituring.com.cn/ 跳 转 到 图 灵 社 区 ， 这 时 就 可 以 使 用 <a> 标 
签 按照 下 面 的 方式 进行 编写 。 
































<a href="http://www.ituring.com.cn/"> 跳 转 到 图 灵 社 区 </a> 





<a> ~ </a> 之 间 编 写 的 是 作为 链接 显示 的 字符 串 。 链 接 的 地 址 要 在 <a> 标签 的 href 属性 中 
指定 。a 和 href 之 间 一 定 要 加 一 个 半角 空格 。 如 果 输 入 了 全 角 空 格 ， 属 性 就 会 无 效 。 
正如 本 例 中 通过 “href=” 设 置 的 URL 一 样 ,设置 为 属性 的 值 称 为 “属性 值 "。 属 性 值 需 要 用 
" "或 1 ' 括 起 来 。 

需要 注意 的 是 ，PHP 中 需要 使 用 print 或 echo 动态 编写 HTML 标签 ， 这 时 ,输出 的 标签 本 
身 也 需要 用 " "或 '' 括 起 来 。 



















































































print "<a href= ~ > 跳 转 到 图 灵 社 区 </a>"; 











在 必须 使 用 两 次 " "或 ' ，' 的 情况 下 ， 如 果 像 下 面 这 样 两 次 使 用 了 相同 的 符号 ， 就 会 发 生 


错误 。 


print "<a href=" http://www.ituring.com.cn/ "> 跳 转 到 图 灵 社 区 </a>"; 














为 了 避免 发 生 错误 ,我 们 需要 采取 一 些 对 策 ， 比 如 在 内 部 的 “"” 前 面 加 上 “\” 进 行 转 义 ; 在 
外 部 用 " " 将 内 部 的 ，，' 括 起 来 ， 或 者 在 外 部 用 ' “将 内 部 的 " " 插 起 来 (一 16.2.2 节 ) 等 。 

如 下 所 示 ， 在 本 书 中 PHP 输出 的 字符 串 会 用 " " 括 起 来 ， 内 部 HTML 标签 的 属性 值 会 用 ' ， 
括 起 来 。 





print "<a href=' http://www.ituring.com.cn/ '> 跳 转 到 图 灵 社 区 </a>"; 


17.4 使 用 PHP 脚本 输出 HTML 文件 


接 下 来 试 着 使 用 PHP 脚本 编写 一 个 点 击 链接 后 可 以 跳 转 到 图 灵 社 区 的 Web 页 面 。<html >、 
<head>、<title>、<body> 等 元 素 都 要 编写 出 来 。 
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代码 清单 17-2 link_page.php 





<?php 

print "<!DOCTYPE html>"; 
print. <html>"; 

print "<head>"; 

print ve<titlesry 

print " 跳 转 到 图 灵 社 区 的 链接 " ; 
DEint Va/Eitlesy 














print "</head>"; 
print "<body>"; 
print "<a href='http://www.ituring.com.cn/'> 跳 转 到 图 灵 社 区 </a>"; 
print "</body>"; 
print, ve/htmls™; 

















?> 











只 需 用 print (或 echo ) 将 标签 输出 即 可 。 我 们 来 实际 执行 一 下 ， 看 看 点 击 显 示 的 网 页 链接 
后 ， 相 应 的 Web 页 面 能 否 显示 出 来 ( 见 图 17-4 )。 
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AU 图 灵 社 区 





es ， 





me 
jz 


em pune EP me ps 


| SQL 进 阶 教程 网 络 是 怎样 连接 的 人 工 智能 简 史 
MICK 户 根 勤 尼克 
周 自 恒 译 


图 17-4 执行 结果 


P 关于 本 书 的 示例 


通过 前 面 的 介绍 我 们 可 以 看 出 ， 在 使 用 PHP 动态 输出 Web 页 面 的 情况 下 ， 必 须 按照 HTML 的 
语法 来 编写 文档 类 型 声明 和 标签 。 但 是 ， 本 书 的 示例 几乎 都 和 下 面 的 PHP 脚本 一 样 没有 编写 标签 。 
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<?php 
print "欢迎 光临 1"; 


> 





之 所 以 没有 编写 标签 ， 是 因为 在 每 个 示例 中 编写 标准 的 HTML 标签 非常 费事 ， 如 果 服 务 顺 端 
的 设置 没有 什么 问题 ， 浏 览 器 都 能 显示 出 正确 的 结果 。 不 过 我 们 只 能 在 练习 的 时 候 这 么 操作 ， 如 果 
是 在 工作 中 创建 Web 应 用 程序 ， 就 不 能 进行 省 略 了 。 

另外 ， 后 面 还 会 介绍 使 用 了 HTML 表单 的 PHP 示例 。 这 部 分 示例 也 只 会 记述 最 低 限度 需要 的 内 容 。 























17.5 需要 记 住 的 标签 


这 里 介绍 一 下 本 书 示 例 中 出 现 的 标签 。 请 大 家 实际 执行 一 下 使 用 了 print 的 PHP 脚本 ， 并 逐 
一 确认 其 执行 结果 。 



































P ”<br> 标签 


<br> 标签 用 于 换行 。 不 管 在 HTML 文件 中 键入 多 少 Enter， 浏 览 需 显示 的 内 容 都 不 会 发 生 改 
变 。 要 想 换行 ， 必 须 使 用 <pr> 标签 。 但 是 ，HTML 5 中 不 能 使 用 <br> 标签 调整 布局 。 
代码 清单 17-3 是 使 用 <br> 标签 的 示例 。 执 行 结果 如 图 17-5 所 示 。 


代码 清单 17-3 opening.php 











<?php 
print " 欢迎 光临 zpr>sQL 咖啡 厅 !"; 














be El 
介 http://localhost/opening.php ”用 搜索 5 司 二 二 | 
@ localhost x 
欢迎 光临 
SQL 咖啡 厅 ! 




















图 17-5 执行 结果 
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区 <hr> 标签 


<hr> 标签 最 初 用 于 创建 水 平 线 ， 但 是 在 HTML 5 中 它 用 于 分 隔 段 落 。 不 过 现在 的 浏览 器 一 般 
都 会 生成 水 平 线 。 
代码 清单 17-4 是 使 用 <hr> 标签 的 示例 。 执 行 结果 如 图 17-6 所 示 。 


代码 清单 17-4 hr.php 








<?php 
print " 欢迎 光临 zpr>sQzL 咖啡 厅 !1"; 
DEint 下 < 


?> 











Ye /cmonor -olsr. 
[Soon “ 国 








PP <img> 标签 
<img> 标签 用 于 显示 图 片 。 具体 格式 如 下 。 
医 天 对 <img> 标签 的 使 用 方法 

















src 属性 中 指定 的 是 要 显示 的 图 片 文件 的 名 称 以 及 存储 的 位 置 。 例 如 ， 代 码 清单 17-5 用 于 显 
示 pic 文件 夹 中 名 为 fathergif 的 图 片 。 执 行 结果 如 图 17-7 所 示 。 


代码 清单 17-5 img.php 




















print "<img src='pic/father.gif' alt=' 爸爸 的 插图 '>"; 
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| S napylocalhosvimgphp 
[iocalhos x] 


|@localthost x*| 





图 17-7 ”执行 结果 
于 是 ，pic 文件 夹 中 的 图 片 father.gif 就 显示 了 出 来 。 
alt 属性 用 于 指定 当 不 能 显示 图 片 时 显示 的 文本 内 容 。 在 HTML 5 中 必须 指定 该 属性 。 在 上 述 
示例 中 ， 如 果 出 于 某 种 原因 图 片 无 法 正常 显示 出 来 ，Web 页 面 上 就 会 显示 “和 爸爸 的 插图 ”。 
































P <meta> 标签 


<meta> 标签 用 于 设置 页 面 信息 。 该 标签 有 很 多 功能 ， 可 以 设置 各 种 各 样 的 信息 ， 比 如 搜索 引 
擎 的 关键 词 、 在 搜索 结果 中 显示 的 文章 、 自 动 跳 转 到 的 页 面 、 字 符 编 码 等 。 

本 节 将 介绍 设置 Web 页 面 字符 编码 的 方法 。 

一 般 来 说 ， 浏 览 器 会 自动 识别 发 送 过 来 的 Web 页 面 的 字符 编码 ， 但 有 时 也 会 出 现 错误 。 如 果 
选择 了 错误 的 字符 编码 ， 就 会 发 生 乱 码 。 大 家 一 定 看 到 过 乱码 的 Web 页 面 。 在 这 种 情况 下 ， 如 果 
使 用 <meta> 标签 告诉 浏览 器 应 该 使 用 什么 样 的 字符 编码 ，Web 页 面 就 能 恢复 正常 了 。 

<meta> 标签 需要 记述 在 <zhead> 和 </head> (一 17.3 节 ) 之 间 ， 具体 格式 如 下 所 示 。 

































































使 用 <meta> 标签 设置 字符 编码 


"字符 编码 " 的 部 分 可 以 按照 表 17-1 中 的 内 容 进行 指定 。 
表 17-1 字符 编码 的 指定 方法 











字符 编码 指定 的 字符 
UTF-8 UTF-8 
GBK GBK 

GB 2312 GB 2312 
GB 18030 GB 18030 
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请 注意 ，UTF-8 中 的 “-” 是 连 字 符 。 
在 指定 字符 编码 为 UTF-8 的 情况 下 ， 代 码 内 容 如 下 所 示 。 





<!DOCTYPE html> 
<html> 
<head> 


<title>meta 元 素 的 例子 </title> 


<meta charset="UTF-8"> 
</head> 





本 书 第 21 章 介 绍 的 “实用 公告 板 ” 的 示例 








FP 使 用 了 这 


文 个 -meta> 标签 。 


17.6 使 用 CSS 指定 颜色 和 字体 大 小 











在 HTML 5 中 需要 使 用 CSS (Cascading Style Sheet， 层 县 样式 表 ) 设置 
小 等 与 样式 相关 的 内 容 。 本 书 只 介绍 需 





[dd = = 











-= 内 
月 时 





和 字符 的 颜色 、 大 





要 用 到 的 各 个 属性 ， 不 会 对 CSS 的 详细 内 容 进行 讲解 。 





5 17.6.1 指定 背景 的 颜色 
CSS 的 设置 方法 中 有 各 种 各 样 的 模式 ， 本 书 采用 了 直接 在 标签 的 style 属性 中 编写 CSS 的 方 
式 。CSS 需要 按照 “属性 : 值 ”这 样 的 语法 来 编写 。 
下 面 是 将 页 面 背景 设置 成 浅 绿色 (aqua ) 的 示例 ， 为 此 我 们 使 用 了 background-color 属性 。 





<body style = 









































style 属 





有 


生 


"background-color :acua'" > 





属性 











代码 清单 17-6 是 执行 上 述 内 容 的 PHP 脚本 。 执 行 结 


代码 清单 17-6 back_col.php 


设置 的 值 


























如 图 17-8 所 示 。 














<?php 





print "<body style = ' background-color:aqua '>"; 








您 htpx/localhostyback col.php 
总 localhost 


x 
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图 17-8 执行 结果 


属性 和 值 之 间 需 要 用 冒号 ( : ) 进行 分 隔 。 不 仅仅 是 background-colot 属性 ，CSS 中 的 颜 














周 值 ) 来 指定 的 ( 见 表 17-2 )。 



























































色 都 是 用 颜色 的 全 称 或 者 RGB 值 (16 进 

表 17-2 style 中 指定 的 颜色 
颜色 全 称 RGB 值 
黑色 black #000000 
海军 蓝 navy #000080 
蓝 色 blue #0000ff 
绿色 green #008000 
蓝 绿色 teal #008080 
黄 绿色 lime #00ff00 
浅 绿色 aqua #OOffff 
褐 红色 maroon #800000 
紫色 purple #800080 
橄榄 色 olive #808000 
灰色 gray #808080 
银色 silver #cOcOcO 
红色 red #ff0000 
紫红 色 fuchsia #ffOOff 
黄色 yellow #ffff00 
白色 white #ffffff 
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5 17.6.2 ”指定 字符 的 大 小 和 颜色 

我 们 可 以 通过 使 用 <aiv> 或 <p> 标签 指定 style 属性 来 设置 整个 段落 的 字符 颜色 和 字体 大 
小 。color 属性 用 于 设置 字符 颜色 ，font -size 属性 用 于 设置 字体 大 小 。<div> 和 <p> 都 用 于 
将 括 起 来 的 范围 定义 为 块 ， 不 过 二 者 的 区 别 是 ,在 <p> 的 情况 下 块 的 前 后 可 以 空 出 一 行 ， 而 
<div> 不 可 以 。 

代码 清单 17-7 使 用 <aiv> 将 字符 串 “ 这 是 SQL 咖啡 厅 的 公告 板 哦 ”设置 为 “ 蓝 色 35 号 字 ”。 
在 这 种 需要 设置 多 个 属性 的 情况 下 ， 我 们 可 以 使 用 分 号 (; ) 将 属性 隔 开 。 执 行 结 果 如 图 17-9 
所 示 。 


代码 清单 17-7 div.php 


























<?php 

print "<div style='color:blue;font-size:35pt'>"; 
print " 这 是 SQL 咖啡 厅 的 公告 板 哦 "; 

print ve/div>"; 


Ve 











MB) S htpy/localhosvydivphp ”用 搜索- 


这 是 SQL 咖 啡 厅 的 公告 板 哦 





图 17-9 执行 结果 
我 们 可 以 使 用 <span> 标签 的 style 属性 来 对 同一 行 的 一 部 分 字符 串 进行 设置 。 代 码 清 
单 17-8 是 将 字符 串 “ 公 告 板 ”设置 为 50 号 字 的 示例 。 执 行 结果 如 图 17-10 所 示 。 








代码 清单 17-8 span.php 





<?php 
print " 这 是 SQL 咖啡 厅 的 <span style='font-size:50pt'> 公告 板 </span> 哦 "; 


党 加 
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17.7 ”Here Document 和 nl2br 函数 


CE hap//ocalhosysp 
x 


NE 
大 localhos 此 
| 这 是 sQL 咖 啡 厅 的 公 口 告 板 。 

















17-10 ”执行 结果 


17.7 Here Document 和 nl2br 函数 


17.7.1 什么 是 Here Document 

使 用 PHP 脚本 编写 Web 页 面 就 意味 着 要 使 用 print 或 echo 将 HTML 标签 作为 文本 输出 。 
使 用 print 等 命令 输出 的 标签 是 字符 串 ， 所 以 需要 用 " " 括 起 来 。 不 过 除 此 之 外 还 有 很 多 麻烦 的 
设置 ， 比 如 需要 用 ' ' 把 属性 值 括 起 来 、 加 上 “\” 等 。 

这 时 ， 能 够 将 大 量 字符 串 存 储 在 变量 中 的 Here Document 功能 就 起 到 非常 大 的 作用 了 。Here 
Document 能 够 将 “< < < 表示 结束 的 字符 ”到 “表示 结束 的 字符 ”之 间 的 内 容 作为 一 连 串 的 字符 串 
进行 处 理 。 

Here Document 









































大 家 不 妨 回想 一 下 17.4 节 介 绍 的 用 于 编写 web 页 面 的 脚本 。 相 信 很 多 人 会 觉得 ， 输 出 HTML 
时 要 输入 好 多 次 print " x x" 实在 是 太 麻 烦 了 。 

如 果 使 用 Here Document 来 编写 这 个 脚本 ， 就 会 变 成 代码 清单 17-9 这 样 的 形式 。“ 表 示 结 束 的 
字符 ”设置 成 了 eot (end of text )。 
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代码 清单 17-9 here.php 





<?php 

$mozi=<<<eot 

<!IDOCTYPE html> 
<html> 

<head> 

<title> 跳 转 到 图 灵 社 
</head> 

<body> 

<a href="http://www.ituring.com.cn/"> 跳 转 到 图 灵 社 区 </a> 
Roy 

= tema 

ect ; 

















区 的 链接 </title> 


7 























Belin no 


Eas 














国 的 部 分 表示 $mozi。 在 此 范围 内 无 须 在 意 是 否 需 要 输入 " "。 回 输出 了 $mozi。 也 就 是 说 ， 
“$mozi=<<<eot” 和 “eot;” 之 间 的 部 分 只 写 HTML 标签 的 内 容 即 可 。 

使 用 print 输出 的 字符 串 必须 用 " "或 ' ' 括 起 来 。 在 用 " "或 ' ' 括 起 来 的 字符 串 中 ， 
如 果 要 将 “"” 或 “'” 作 为 字符 输出 ， 就 需要 进行 特殊 人 处理 ， 比 如 加 上 “\” 等 。 而 在 Here 
Document 中 则 不 存在 这 样 的 问题 。 编 写 " "或 ' ' 较 多 的 SQL 语句 和 HTML 标签 时 ，Here 
Document 会 起 到 非常 大 的 作用 。 





1772 什么 是 nzbr 画 数 


那么 ， 代 码 清单 17-10 的 PHP 脚本 会 如 何 显示 呢 ? 


代码 清单 17-10 nl2br_nasi.php 





<?php 

$str=<<<eot 

你 好 
晚上 好 

ect ; 

PELNt et; 


人 











“你 好 ”“ 晚 上 好 ”这 两 个 字符 品 作 为 Here Document 赋 给 了 变量 str， 最 后 通过 print 输出 。 
在 国 中 , “你 好 ”和 “晚上 好 ”用 两 行 表 示 。 但 是 ， 执 行 这 个 代码 后 ，Web 页 面 会 像 图 17-11 这 
样 将 “你 好 ”和 “晚上 好 ”显示 在 一 行 中 。 
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| 
eww -els 
Ek 
你 好 晚上 好 





17-11 执行 结果 

字符 串 换行 意味 着 字符 串 中 包含 了 换行 符 。 但 是 ， 如 果 没 有 <br> (一 17.5 节 ) 等 用 于 换行 的 
标签 ，Web 页 面 中 显示 的 内 容 就 不 会 换行 。 

例如 可 能 存在 这 样 的 问题 。 假 设 在 公告 板 的 输入 对 话 框 中 输入 了 已 经 换行 的 信息 。 如 果 直 接 用 
print 输出 这 个 信息 ， 这 个 信息 就 会 变 成 相连 的 字符 串 。 

在 这 种 情况 下 如 果 忽 略 了 换行 ， 就 可 能 会 遇 到 麻烦 。 这 时 就 需要 使 用 mn12bz 函数 。 该 函数 具 
有 在 字符 串 中 的 每 个 新 行 之 前 插入 换行 符 的 功能 。 

代码 清单 17-11 用 n12br 函数 对 $stzt 的 值 进行 了 处 理 ， 并 通过 print 输出 处 理 后 的 值 ， 也 
就 是 将 “print $str;” 的 部 分 换 成 了 “print nl2br ($str ) ;”。 执行 结果 如 图 17-12 所 示 。 























代码 清单 17-11 nl2br.php 





<?php 

$str=<<<eot 

你 好 

晚上 好 

OL 

Stine T1200 


草包 








ewearnoe ol 
EEETEEEE 区 = 


你 好 


晚上 好 





17-12 ”执行 结果 
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由 此 便 实 现 了 换行 功能 。 
FE 翅 下 <textarea> 标签 


标签 -textarea> ~ </textarea> 用 于 设置 可 输入 多 行 字 符 串 的 文本 区 域 (一 21.4.1 节 )。 当 对 输入 到 
该 标签 内 的 字符 捉 执 行 换行 处 理 时 ，n12br 函数 就 会 显示 出 它 的 威力 。 






































17.8 ”使 用 PHP 从 浏览 器 发 送 和 接收 数据 


掌握 了 所 有 的 基础 知识 之 后 ， 我 们 来 学 习 一 下 使 用 PHP 从 浏览 器 传输 数据 的 步骤 吧 。 

















局 17:8.1 浏览 器 和 PHP 文 件 之 间 的 数据 交换 。 


首先 来 看 一 下 当 从 浏览 器 操作 MySQL 时 ， 数 据 是 如 何 交 换 的 ( 见 图 17-13 )。 数 据 交 换 的 具体 步 
又 如 下 所 示 。 














QD 用 户 从 浏览 器 显示 的 Web 页 面 上 发 送 数据 。 














a 





@ Web 服务 器 上 的 PHP 文件 接收 数据 。 





@ PHP 脚本 与 MySQL 数据 库 进 行 交 互 。 


a 





由 PHP 脚本 输出 包含 结果 的 Web 页 面 ， 并 将 其 发 送 回 浏览 





@) 浏览 器 接收 结 




















17-13 浏览 器 和 PHP 和 MySQL 








在 使 用 MySQL 之 前 ， 我 们 先 来 学 习 一 下 HIML 文件 和 PHP 脚本 之 间 进 行 数据 交换 的 步骤 。 
试 着 创建 如 下 机 制 。 
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P 数据 的 发 送 方 

2 HTML 文件 send.html 

2 通过 点 击发 送 按钮 来 发 送 输入 到 文本 框 中 的 字符 串 
P 数据 的 接收 方 


2 PHP 文件 receive.php 
人 按照 原样 显示 接收 的 数据 














send.html 是 发 送 方 的 Web 页 面 (HTML 文件 )，receive.php 是 接收 方 的 PHP 脚本 。 这 是 一 个 


用 receive.php 显示 send.html 发 送 的 数据 的 示例 〈 见 图 17-14 )。 两 个 文件 都 保存 在 Web 服务 器 发 布 
的 文件 夹 中 。 





send.html /NN 














17-14 在 文件 之 间 发 送 和 接收 数据 


下 一 节 我 们 将 制作 一 个 从 Web 页 面 发 送 数据 的 HTML 文件 ( send.html )， 然 后 在 17.8.3 节 创 建 
作为 接收 方 的 PHP 脚本 ( receive.php )。 





17.8.2 ”制作 一 个 用 于 发 送 数据 的 Web 页 面 send.html 


OT sn 一 
首先 来 制作 从 Web 页 面 发 送 数据 的 HTML 文件 。 这 里 要 建立 如 下 机 利 





2 


o 


2 点 击 按钮 ， 文 本 框 中 输入 的 数据 就 会 发 送出 去 


需要 用 到 的 组 件 有 两 个 ， 即 “按钮 ”和 “文本 框 ”"。 这 两 个 组 件 由 后 面 介绍 的 <input > 标签 来 
设置 。 当 创建 某 发 送 机 制 时 ， 一 般 需 要 创建 一 个 表单 ， 然 后 在 其 中 设置 文本 框 和 按钮 等 组 件 。 
创建 如 代码 清单 17-12 所 示 的 文本 文件 ， 并 将 这 个 文本 文件 以 send.html 为 文件 名 保存 在 发 布 的 




















390 | 第 17 章 PHP 脚本 和 HTML 


文件 夹 中 。 这 里 省 略 了 <html>、<head>、<body> 之 类 的 标签 。 


代码 清单 17-12 send.html 





<form method="POST" action="receive.php"> 
<input type="text" name="a"> 

<div> 

<input type="submit" value=" 发 送 "> 

</div> 


</form> 











那么 ,文件 是 否 以 send.html 为 文件 名 保存 到 了 发 布 的 文件 夹 中 呢 ? 确认 Apache 正在 运行 且 
编写 的 脚本 没有 错误 后 ， 在 浏览 器 的 地 址 栏 内 输入 http://localhost/send.html。Web 页 面 应 该 会 像 
图 17-15 这 样 显示 出 来 。 





je 一 name="a' 的 文本 框 
Ea 





























action="receive. 
全 phpmethod=" POST 
的 表单 

















type="submit 的 按钮 














图 17-15 send.html 
表单 成 功 显示 后 ， 我 们 来 看 一 下 各 标签 的 功能 。 


PP <form> 标签 


<form> 标签 用 于 创建 表单 。 使 用 <form> 标签 创建 一 个 表单 ， 并 在 其 中 使 用 <input > 等 标 
签 设 置 文本 框 和 按钮 。<form> 标签 的 格式 如 下 所 示 。 


<form> 标签 的 格式 


i <form method=" 发 送 方法 " action=" 目标 地 址 "> 









































在 这 里 编写 <input> 等 标签 ,设置 按钮 和 文本 框 


ooooooooeeesssoooseoooe ooo esos ooo ooo eros tos eee ooo ooo 0s 000 0】 0】 0】 0 00000 0000. 
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<form> 标签 中 可 以 设置 action 属性 和 methogd 属性 ( 表 17-3 )。 


表 17-3 <form> 标签 中 设置 的 主要 属性 





























属性 内 容 
action 指定 发 送 数据 的 目标 地 址 ， 即 指定 接收 和 处 理 数 据 的 程序 
method 指定 发 送 数据 的 方法 (一 17.9.1 节 )。 有 POST 和 GET 两 种 方法 























除了 action 和 method 以 外 ，<form> 标签 中 还 可 以 设置 用 于 指定 发 送 方法 (MIME 类 型 ) 
的 enctype 属性 等 。 





区 <input> 标签 


通过 在 <form x x > 和 </form> 之 间 编 写 <input> 标签 ,我 们 可 以 设置 用 于 输入 发 送 数据 
的 文本 框 ， 以 及 发 送 按钮 等 组 件 ( 见 表 17-4 )。 


< input> 标签 的 格式 


; <input type=" 按钮 的 种 类 " name=" 用 于 识别 数据 的 元 素 名 称 "， size=" 大 小 " 
value=" 显示 的 字符 " 和 " 发 送 的 值 " " 默认 值 "> 


























表 17-4 <input> 标签 中 可 以 设置 的 主要 属性 






























































属性 内 容 

type 编写 用 于 指定 组 件 种 类 的 字符 串 

name 设置 要 附加 到 组 件 的 元 素 名 称 ， 该 名 称 用 于 识别 数据 
size 设置 文本 框 的 宽度 

value 当 字 符 串 显 示 在 按钮 等 地 方 时 ， 设 置 字符 串 内 容 









































type 属性 中 可 以 设置 表 17-5 中 列举 的 组 件 。 











表 17-5 type 属性 的 设置 








描述 可 设置 的 组 件 种 类 
type="submit" 可 以 发 送 数据 的 按钮 
type="button" 按钮 

type="text" 文本 框 





type="checkbox" 复 选 框 





type="radio" 单 选 按钮 
type="hidden" 隐藏 ( 仅 发 送 数 据 ) 
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BP 使 用 <input> 标签 的 示例 


代码 清单 17-13 是 使 用 <input > 标签 的 示例 。 这 是 一 个 元 素 名 称 为 al 且 “ 大 小 ”为 50 的 文 
本 框 。 执 行 结 果 如 图 17-16 所 示 。 


代码 清单 17-13 input_text.html 











<input type="text" name="al" size="50"> 











ER 
YO weecatosvnou sntin -ol 

















17-16 ”执行 结果 
代码 清单 17-14 设置 的 是 显示 为 “点 击 ” 的 发 送 按 钮 。 执 行 结果 如 图 17-17 所 示 。 


代码 清单 17-14 input_submit.html 











<input type="submit" value=" 点击 "> 








le 
< ET - E 


点 击 

















17-17 ”执行 结果 
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代码 清单 17-15 设置 的 是 一 个 元 素 名 称 为 ri1、 发 送 的 值 为 pad 的 单 选 按钮 。 执 行 结果 如 
图 17-18 所 示 。 


代码 清单 17-15 input_radio.html 





<input type="radio" name="r1" value="bad"> 











OW 


€ 和 @ http://localhost/input_radio.html = © FA WR 





17-18 执行 结果 
执行 代码 清单 17-16 中 的 代码 后 ， 浏 览 品 上 不 会 显示 任何 内 容 ， 但 会 发 送 字符 “隐藏 数据 "。 


代码 清单 17-16 hidden.html 





<input type="hidden" value=" 隐藏 数据 "> 











设置 了 type="hidaen" 的 <input> 标签 不 会 在 浏览 器 上 显示 任何 内 容 。 该 功能 主要 用 于 强 
制 发 送 不 需要 向 用 户 显示 的 数据 (一 21.4.1 节 )。 


> 确认 创建 的 表单 
我 们 来 确认 一 下 代码 清单 17-12 的 内 容 。 
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P <form method="POST" action="receive.php"> 


<form> 标签 中 添加 了 action="receive.php"。 因 此 ， 点 击 “ 发 送 ” 按 钮 ， 就 会 将 数据 发 
送 给 名 为 receive.php 的 PHP 
因为 method 属性 中 指定 了 “POST”， 所 以 会 采用 POST 方法 发 送 数据 。POST 发 送 方法 的 相 





关内 容 会 在 17.9 节 介 绍 。 


Pp <input type="text" name="a"> 


因为 type 属性 为 text， 所 以 会 创建 一 个 文本 框 ( 见 表 17-5 )。name 属性 为 a， 因 此 接收 数 
据 的 PHP 脚本 receive.php 会 以 a 这 个 名 字 来 识别 数据 。 











戎 <input type="submit" value=" 发 送 "> 


因为 type 属性 是 supmit， 所 以 设置 的 是 可 以 发 送 数据 的 按钮 。 此 外 ，value 属性 为 “发 
送 ”， 因 此 按钮 上 会 显示 发 送 这 个 词 。 














@17-8.3 创建 receive.php 以 接收 和 显示 数据 


这 次 我 们 来 创建 用 于 接收 数据 的 receive.php。receive.php 用 于 接收 在 send.html 的 文本 框 中 输入 
的 数据 。 
因为 在 send.html 的 <form> 标签 中 ，method 属性 被 设置 为 了 “PoSsT”， 所 以 会 采用 POST 方 
法 发 送 数据 。 采 用 POST 方法 发 送 的 数据 会 被 变量 $_POST 接收 。 变 量 $_POST 的 具体 格式 如 下 。 


定名 里 变 量 $_ POST 保存 method="POST" 发 送 的 数据 












































$_POST 是 PHP 中 预定 义 的 变量 名 。POST 发 送 的 数据 会 作为 数组 保存 在 这 个 变量 中 。 

$_POST 可 以 当 作 关联 数组 (一 16.7.3 节 ) 使 用 。 数 组 下 标 指定 用 于 识别 数据 的 元 素 名 称 。 

这 里 文本 框 中 附加 了 a 这 个 用 于 识别 数据 的 元 素 名 称 (name="a" )。 因此， 在 receive.php 中 ， 
如 果 指 定 "a" 为 关联 数组 的 下 标 ， 那么 只 要 将 变量 编写 为 $_POST [ "a" ]， 就 可 以 接收 发 送 过 来 
的 数据 了 。 


接收 输入 到 name="a" 的 文本 框 中 的 值 
































代码 清单 17-17 是 将 send.html 发 送 过 来 的 数据 直接 输出 的 receive.php。 创 建 receive.php， 并 把 
它 保存 到 公布 的 文件 夹 中 。 
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代码 清单 17-17 receive.php 





<?php 
ee eS a 


?> 











贺 表 示 通 过 元 素 名 称 来 取出 接收 的 值 。 


超级 全 局 变量 
像 $ posT 这 种 不 需要 进行 声明 ， 可 以 用 在 程序 任何 地 方 的 变量 称 为 “超级 全 局 变量 "。GET 方法 
(一 17.9.2 节 ) 中 使 用 的 $_GET、 使 用 Cookie 时 会 用 到 的 $_cooKIE 等 都 属于 超级 全 局 变量 。 































































































@)17:8.4 发 送 和 接收 数据 

我 们 来 实际 发 送 和 接收 一 下 数据 。 实 际 操作 之 前 要 确认 17.8.2 市 的 send.html 和 17.8.3 节 的 
receive.php 已 经 保存 在 了 发 布 的 文件 夹 中 。 另 外 ， 确 认 Apache 处 于 运行 状态 (一 2.2.3 节 )。 

如 果 在 浏览 器 的 地 址 栏 中 输入 http://localhost/send.html，send.html 就 会 像 17.8.2 节 介 绍 的 那样 
显示 出 来 。 在 文本 框 中 输入 要 发 送 的 字符 串 〈 此 处 输入 的 是 “发 送 字符 串 ”)， 点 击 “ 发 送 ”按钮 。 
于 是 ， 数 据 就 会 传递 给 receive.php， 输 入 的 字符 串 就 会 显示 出 来 ( 见 图 17-19 )。 





























17-19 执行 结果 


在 同一 台 计算 机 上 执行 上 述 操作 可 能 无 法 深切 感受 到 数据 发 送 与 接收 的 情况 ， 不 过 在 网 络 上 发 
送 和 接收 数据 ， 使 用 的 方法 基本 也 是 一 样 的 。 
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17.9 通过 POST 和 GET 发 送 数据 


在 17.8.2 节 ， 我 们 采用 了 在 <form> 标签 中 编写 method="POST" 的 方法 把 数据 从 Web 页 盏 
传递 给 PHP 脚本 。 这 种 传递 数据 的 方式 称 为 POST。POST 在 前 面 已 经 出 现 了 好 几 次 。 
在 传递 数据 的 方法 中 ， 除 了 POST 之 外 还 有 GET。POST 和 GET 的 特征 如 下 所 示 。 

















Pb POST 

3 数据 不 在 URL 中 

3 可 以 发 送 文本 或 二 进 制 格式 的 数据 
Pp GET 


2 将 数据 添加 到 URL 中 发 送 ( 数据 可 见 。 可 能 会 发 送 非法 数据 ) 
2 只 可 以 发 送 文本 格式 的 数据 
2 如 果 没 有 进行 任何 声明 ， 将 使 用 GET 方法 ( 默认 ) 








下 面 我 们 就 来 看 一 看 POST 和 GET 之 间 具 体 有 哪些 差异 。 











如 果 想 使 用 GET 方法 发 送 数据 ， 以 上 一 节 创 建 的 send.html 和 receive.php 为 例 ， 将 列表 中 记载 
的 字符 “POST” 和 替换 为 “GET”， 发 送 方法 就 变 成 了 GET。 

试 着 将 send.html 和 receive.php 的 发 送 方法 改 为 GET。 代 码 清单 17-18 将 接收 方 receive.php 中 
的 $ PosT [ "a" |] 的 部 分 改 成 了 $_GET | "a" ] ( 国 )。 这 样 修改 之 后 ， 就 可 以 接收 由 GET 发送 来 
的 name="an 的 数据 。 我 们 把 修改 后 的 PHP 文件 的 名 称 设置 为 get_ receive.php。 


代码 清单 17-18 get_receive.php 























<?php 
Brinee :emia 


傈 名 











代码 清单 17-19 的 脚本 get_send.html 将 发 送 方 send.html 中 的 method="POST" 部 分 改 为 了 
method="GET" ( 回 ), 男 外 ， 因 为 接收 方 的 PHP 文件 的 名 称 为 get_receive.php， 所 以 将 
action="receive.php" 改 为 了 action="get receive.php"。 执 行 结果 如 图 17-20 所 示 。 
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代码 清单 17-19 get_send.html 





EeegmeenegE RS at len tece ee 
<input type="text" name="a"> 

<div> 

<input type="submit" value=" 发 送 "> 

</div> 

</form> 











OX 
| httpi//ocalhost/get receivephp?a=%eE5%68F3691%6E936803%68 ~ O 从 六 
1 








17-20 ”执行 结果 


a17.9.3 GET 和 POST 的 F 别 


使 用 GET 方法 发 送 数据 和 显示 结果 ( 见 图 17-22 ) 与 使 用 POST 方法 的 情况 ( 见 图 17-21 ) 几 
乎 完全 相同 ， 但 仅 有 一 处 不 同 。 请 仔细 观察 这 一 处 在 哪里 。 





CD | hes/ocalhostreceve.php ES 





发 送 字符 串 





17-21 使 用 POST 方法 发 送 数据 的 情况 






ME)| ES http://ocalhost/get receive.phpba=3tE5968F3691%6E93680368196E5s6ADs69796E796ACs6A6%E4%6BBsB2| -0o| 
名 localhost x 








数据 添加 到 了 这 里 














17-22 ”使 用 GET 方法 发 送 数据 的 情况 
发 现 二 者 的 区 别 了 吗 ? 实际 上 ， 当 使 用 GET 方法 发 送 数据 时 ，URL 后 面 会 显示 ?a=%E5%8F&91 
之 类 的 内 容 "。 而 当 使 用 POST 方法 发 送 数据 时 ,地 址 栏 上 只 会 显示 http://localhost/receive.php。 也 就 





四 在 Chrome 或 者 FireFox 等 浏览 器 上 显示 的 内 容 可 能 会 有 所 不 同 。 





译 者 注 
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是 说 ， 在 使 用 GET 方法 的 情况 下 ， 数 据 是 作为 URL 的 一 部 分 发 送 的 。 
当 使 用 GET 方法 时 ， 显 示 在 地 址 栏 上 的 内 容 如 下 所 示 。 
当 使 用 GET 方法 时 地 址 栏 上 显示 的 内 容 








“数据 名 称 ” 是 name ="a" 等 附加 在 文本 框 上 的 “用 于 识别 数据 的 元 素 名 称 "。 如 果 发 送 的 是 
中 文 等 需要 占 2 个 字 节 的 数据 ， 会 先 将 其 转换 为 其 他 代码 ， 然 后 发 送 。 

因此 ， 在 使 用 GET 方法 发 送 数据 的 情况 下 会 看 到 “要 发 送 的 数据 ”和 “用 于 识别 数据 的 元 素 
名 称 "”。 恶 意 攻击 者 可 能 会 把 非法 数据 附加 到 URL 上 发 送 或 偷 看 数据 。 在 使 用 GET 方法 构建 系统 
的 情况 下 ， 必 须 充分 考虑 安全 方面 的 问题 。 











17.9.4 试 着 用 GET 方法 将 值 添 加 在 URL 上 发 送 


前 面 介绍 了 在 使 用 GET 方法 发 送 数据 的 情况 下 ， 数 据 会 添加 到 URL 上 发 送 。 也 就 是 说 ， 即 使 
不 使 用 HTML 文件 ， 只 要 将 数据 添加 在 URL 上 ， 就 可 以 通过 GET 方法 进行 通信 。 

下 面 来 使 用 GET 方法 发 送 数 据 。 试 着 使 用 17.9.2 节 的 get_send.html 发 送 数 据 “12345”。 请 将 
下 面 的 字符 串 输入 地 址 栏 ， 然 后 按 Enter 键 。 执 行 结 果 如 图 17-23 所 示 。 























http://localhost/get receive.php?a=12345 


i 
必 | EG http://ocalhosy/get receivephp?a=12345 














图 17-23 ”执行 结果 
可 以 看 到 ，Web 页 面 上 正确 显示 出 了 “12345”。 
发 送 的 数据 也 可 以 是 中 文 。 这 次 试 着 使 用 method="GET" 发 送 字 符 串 “练习 ”。 在 UTF-8 中 ， 
“ 练 ”的 字符 编码 是 “E7 BB 83”,“ 习 ”的 字符 编码 是 “E4 B9 A0”。 因 此 ， 在 设置 为 name="a" 的 
文本 框 中 输入 “练习 ”， 然 后 执行 submit ， 就 等 于 发 送 了 下 面 的 URL。 执 行 结果 如 图 17-24 所 示 。 




































































http://localhost/get receive.php?a=%E7%BB%83%E4%B9%AO 
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17-24 ”执行 结果 
sa 17.9.5 “在 不 进行 任何 声明 的 情况 下 发 送 数据 
如 果 不 进 行 任何 声明 ， 会 自动 使 用 GET 方法 发 送 数据 。 
例如 ， 试 着 创建 一 个 没有 对 <fozm> 标签 进行 设置 ， 仅 会 显示 一 个 超 链接 的 文件 (代码 清 
单 17-20 )。 执行 结果 如 图 17-25 所 示 。 


代码 清单 17-20 get_send2.html 








<a href="get receive2.php?a=777"> 发 送 777</a>; 











Tx 


和 EEC Te 人 


优 localhost 0 


| 发 送 777: 

















17-25 ”执行 结果 
然后 ， 用 代码 清单 17-21 的 文件 接收 此 信息 。 执 行 结果 如 图 17-26 所 示 。 


代码 清单 17-21 get_receive2.php 








<?php 
print $ GETI"ar); 


?> 











[LEE > 
MBS pocahoseetreceivezphp?a-777 =ojlas. 2- 
[Scenen “站 


777 | 





17-26 ”执行 结果 
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只 需 点 击 get_send2.html 的 超 链 接 ， 即 可 发 送 和 接收 “777”， 并 显示 出 “777”。 也 就 是 说 ， 不 
需要 使 用 method="GET" 等 声明 ， 数 据 就 能 发 送出 去 。 
在 使 用 GET 方法 的 情况 下 ， 我 们 可 以 像 上 面 那样 轻松 地 将 数据 添加 到 超 链接 中 发 送 。 


Google 是 否 也 可 以 把 数据 添加 在 URL 上 

当 使 用 Yahool 或 Google 等 搜索 引擎 进行 搜索 时 ， 相 信 很 多 读者 发 现 了 地 址 栏 的 URL 后 面 跟着 奇怪 的 字符 
串 。 其 实 ， 这 么 做 的 目的 是 将 字符 串 添 加 到 URL 上 来 发 送 数据 。 
在 2017 年 6 月 "的 现在 ， 如 果 在 地 址 栏 中 输入 以 下 内 容 并 按 Enter 键 ， 得 到 的 搜索 结果 就 与 在 Google 搜索 
口中 输入 “abcde” 的 结果 相同 。 
的 内 容 用 于 在 Google 上 搜索 字符 串 “abcde”“?。 执 行 结果 如 图 17-27 所 示 。 


人 eeleepE= ereliyA=i= hael le TlelelelS 


文 里 输入 http://www.google.com/search?q=abcde 
三 







































































































































































找到 约 10.900,000 条 结果 ( 用 时 0.33 秒 ) 


视频 


Urple Teaches the ABCDE & 12345 ABCDE Song 
Letters ABCDE Leam with Kids 
Alphabet Video Rainbow ABC | 

Learmning 


Alphabet ... 
DJC Kids Kids Rainbow ABC LetsStartSmart 


17-27 ”执行 结果 


如 果 是 半角 字符 ， 只 要 把 想 搜 索 的 字符 放 在 “?q=x x” 的 x x 中 即 可 。 请 大 家 多 多 尝试 。 
另外 ， 在 Yahoo ! 中 进行 搜索 的 方法 如 下 所 示 。 这 里 使 用 的 关键 字 是 “p"。 执 行 结果 如 图 17-28 所 示 。 


https://search.yahoo.com/search?p=abcdqe 

































































中 这 里 指 日 文 原版 书写 作 的 时 间 。 在 2018 年 9 月 ， 结 果 也 没有 发 生 改 变 。 
© 内 用 户 可 能 无 法 访问 Google 搜索 引擎 ， 可 尝试 使 用 百度 。 一 译 者 注 





译 者 注 
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@ ET -9c nA 
[ERA 且 当 








Signin 自 


- 在 这 里 输入 https://search.yahoo.com/search?p=abcde 


























Also try: abcde of skin cancer, abcde bundle. abcde melanoma. abcde name 


AbCdE - YouTube@ 

www.youtube.com/channel/UChYsCFeE3Iwr2znq3H3xArw x 

Hi everyone. Welcome to AbCdE. Here at AbCdE we believe that fun and games help kids leam and 
grow. By using Kinder Surprise, Surprise Eggs, Shopkins, Lego, trains and Nursery Rhymes we 
believe 


ABCDE - What does ABCDE stand for? The Free Dictionary@ 
acronyms .thefreedictionary.com/ABCDE v 

‘Want to thank TFD for its existence? Tell a friend about us, add a link to this page. or visit the 
Webmasters page for free fun content. Link to this page: 


Abcde - A Better CD Encoder@ 

Ily.org/~rew/abcde/page ~ 

A Better CD Encoder Abcde is a frontend command-line utility (actually, ashell script) that grabs 
tracks off a CD. encodes them to 099g or mp3 format, and tags them, all in one go. 








图 17-28 执行 结果 
的 方法 来 搜索 中 文字 符 。 








另外 ， 我 们 也 可 以 利用 上 


























17.10 总 结 


本 章 介 绍 了 以 下 内 容 。 


@ 动 态 Web 页 面 和 静态 Web 页 面 的 区 别 

@ HTML 基础 标签 的 编写 方法 

@ 从 Web 页 面 发 送 数据 的 方法 ， 以 及 PHP 脚本 中 的 处 理 内 容 及 流程 
@ 创 建 表单 发 送 数据 的 方法 

@ 数 据 的 发 送 方法 


Web 应 用 程序 的 基础 知识 不 仅 包括 PHP， 还 包括 POST 和 GET。 我们 要 牢记 POST 和 GET 之 
间 的 区 别 。 





> 自我 检查 
下 面 检查 一 下 本 章 学 习 的 内 容 是 否 全 部 理解 并 掌握 了 。 


口 了 解 HTML 的 基本 编写 方法 
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口 能 够 理解 <pr>、<hr>、<a>、<img>、<meta>、<div>、<p> 和 <span> 等 标签 的 含义 
口 能 够 使 用 <form> 标签 创建 表单 

口 能 够 理解 <input> 标签 的 使 用 方法 

口 能 够 使 用 POST 方法 发 送 文本 框 的 值 

口 能 够 使 用 GET 方法 发 送 数据 

口 能 够 理解 POST 和 GET 的 区 别 





园 练习 题 





问题 1 




















请 创建 “从 100 个 单 选 按钮 中 选择 并 发 送 年 龄 数据 ”的 表单 radio.php ( 图 17-29 ) 和 “接收 数据 并 显 
示 包 含 了 年 龄 信息 ”的 PHP 脚本 radio_receive.php ( 图 17-30 )。 



































@ ) @ http://localhost/radio.pht PD ~ © | 愿 Iocalhost x | 而 | { Y 
请 选择 您 的 年 龄 ， 然 后 点 击 “发 送 "按钮 。 
ol1020304050607 O8 09Ol10 

O11 O12 O13 O14 O15 O16 O17 O18 O19 O20 
O21 O22 O23 O24 @25 O26 O27 O28 O29 O30 
O31 O32 O33 O34 O35 O36 O37 O38 O39 O40 
O41042 043 DO44 045 D46 047 048 O49 050 
O51 O52 O53 O54 O55 O56 O57 O58 O59 O60 
O61 O62 O63 O64 O65 O66 O67 O68 O69 O70 
O71 O72 O73 O74 O75 O76 O77 O78 O79 O80 
O81 O82 O83 O84 O85 O86 O87 O88 O89 O90 
O91 O92 O93 O94 O95 O96 O97 O98 O99 O100 
| 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 选择 并 发 送 25 的 示例 












































17-29 radio.php 


站 一口 | x 
【¢ | Er -colss. 2- 
[rer To > 


ET 
接收 了 25 这 个 数据 的 示例 




















17-30 radio_receive.php 








制作 100 个 单 选 按钮 是 重点 。 实 现 一 行 10 个 按钮 的 方法 有 很 多 ， 使 用 while 和 if 会 比 

















较 容易 理解 。 
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国 参考 答案 








问题 1 








创建 代码 清单 17-22 和 代码 清单 17-23 中 的 PHP 脚本 。 
代码 清单 17-22 示例 radio.php ( 用 于 发 送 表 单数 据 的 PHP 脚本 ) 





<form method="POST" action="radio receive.php"> 
请 选择 您 的 年 龄 ， 然 后 点 击 “ 发 送 ” 按 钮 。<br> 

















<?php 


$i=1; 

$c=1; 

beint "ediv 

while ($i<=100){ 
print "<input type='radio' name='r' value='$i'>$i "; 
(SG==10)4 

print "</div><div>"; 
$c=0; 

} 

$i++; 

$C++; 


} 


?> 


<input type="submit" value=" 发 送 "> 
</div> 
</form> 











代码 清单 17-23 ”示例 radio_receive.php ( 用 于 接收 表单 数据 的 PHP 脚本 ) 





<?php 
print " 您 的 年 龄 是 ".$ POST["r"] ." 岁 "; 


?> 














第 18 章 ”使 用 PHP 脚 本 操作 MySOL 


从 本 章 起 ， 我 们 要 开始 使 用 PHP 脚本 操作 MySQL 数据 库 
了 。 那 么 如 何 使 用 PHP 连接 MySQL 服务 器 呢 ? 本 章 会 先 介 绍 
PHP 和 MySQL 的 连接 方法 ， 然 后 介绍 如 何 通 过 PHP 执行 SOL 
语句 。 让 我 们 一 起 努力 学 习 吧 | 













































































18.1 使 用 PHP 脚本 连接 到 MySOQL 服务 器 


S 18.1.1 从 PHP 连接 到 数据 库 的 方法 
使 用 PHP 操作 MySQL 的 方法 有 很 多 。 在 编写 本 书 的 时 候 ，PHP 的 最 新 版 本 是 PHP 7。 在 PHP 7 
中 我 们 可 以 使 用 mysq1li 函数 (mysqli 类 )， 或 者 使 用 PDo 类 来 操作 MySQL。 

PHP 支持 许多 种 数据 库 。 但 是 ， 看 到 mysqgqli 中 的 “mysql” 我 们 就 能 明白 ,，mysqli 仅 
支持 MySQL。 一 般 来 说 ,，mysqli 比 PDO 的 处 理 速 度 快 。 但是, 使 用 了 mysqli 的 程序 无 法 
操作 MySQL 以 外 的 数据 库 。 对 我 们 这 样 的 一 般 用 户 来 说 ， 在 程序 功能 保持 不 变 的 情况 下 只 
改 数据 库 种 类 的 情况 也 偶 有 发 生 。 所 以 理想 的 状态 是 ， 即 使 改变 数据 库 的 种 类 ， 程 序 也 能 正常 
运行 。 

如 果 使 用 PDO (PHP Data Object )， 那 么 无 论 哪 种 数据 库 ， 都 可 以 使 用 相同 的 PHP 脚本。 因此 ， 
对 于 使 用 PHP 操作 MySQL 服务 器 的 相关 内 容 ， 本 书 会 介绍 “使 用 PDo 操作 MySQL” 这 一 更 为 通 
用 的 方法 。 
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mysql 函数 

作为 使 用 了 PHP 和 MySQL 的 应 用 程序 示例 ， 很 多 图 书 和 网 站 介绍 了 使 用 mysql_connect () 和 mysal 
query () 等 函数 连接 数据 库 的 方法 。 所 谓 使 用 mysql 函数 连接 数据 库 ， 其 实 就 是 把 SQL 语句 传递 给 MySOL， 
不 难 理解 。 但 是 ，PHP 从 版 本 5.5 开始 就 不 推荐 使 用 mysql 函数 了 ， 现 在 "发 行 的 PHP 7 已 经 不 能 使 
mysql 函数 了 。 













































































































































































18.1.2 什么 是 PDO? 什么 是 类 ? 什么 是 方法 ? 


PDO 是 PHP 5.1.0 及 更 高 版 本 中 定义 的 标准 数据 库 连 接 机 制 ， 其 目的 是 更 方便 地 利用 数据 库 。 
PDO 是 用 于 管理 数据 库 的 “类 ”( class )， 通过 其 “对 象 ”( object ) 来 操作 数据 库 。 不 过 本 书 只 会 介 
绍 使 用 PHP 操作 MySQL 所 需要 的 基础 知识 和 具体 的 操作 方法 ， 不 会 对 类 和 对 象 进行 详细 介绍 。 如 
果 想 了 解 PHP 面向 对 象 方面 更 多 的 信息 ， 可 以 参考 PHP 的 专业 图 书 。 

那么 ， 类 又 是 什么 呢 ? 虽然 类 有 各 种 各 样 的 解释 ， 但 请 把 类 当 作 数据 和 操作 的 集合 。 也 就 是 
说 ，PDo 类 中 包含 了 许多 用 于 处 理 数 据 库 的 数据 和 操作 方法 。 

只 是 一 个 “定义 ”。 要 进行 实际 操作 ， 必 须 创 建 特定 的 操作 对 象 。 通 过 类 创建 的 特定 操作 对 
象 称 为 对 象 。 例 如 ， 在 操作 MySQL 服务 器 的 情况 下 ， 就 需要 先 通过 PDO 类 创建 一 个 用 于 操作 数据 
库 的 PDO 对 象 ( 见 图 18-1 )。 












































PDO 类 





PDO { 
public bool beginTransaction ( void ) 
public bool commit ( void ) 
public mixed errorCode ( void ) 


public PDOStatement query ( string $statement ) 
public bool rollBack ( void ) 
public bool setAttribute ( int $attribute , mixed $value ) 





图 18-1 对 象 和 方法 
而 方法 是 指 从 属于 类 的 函数 。 前 面 提 到 了 好 几 个 PHP 函数 ， 这 些 函 数 都 是 可 以 随时 使 用 的 。 
但 方法 只 能 在 创建 了 对 象 的 情况 下 才能 使 用 ， 和 函数 的 描述 方式 也 有 些许 不 同 。 























18.1.3 ”使 用 PDO 连接 到 MySQL 服务 器 


下 面 我 们 就 来 使 用 PDO 操作 MySQL 服务 器 。 要 使 用 PDO 类 ， 就 得 先 创 建 一 个 具体 的 PDO 对 象 。 
创建 对 象 的 方法 有 很 多 种 ， 这 里 我 们 来 介绍 使 用 命令 new 创建 对 象 的 方法 ， 具体 语 法 如 下 所 示 。 


中 指 编写 日 文 原 书 的 时 间 。 















































译 者 注 
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创建 PDO 类 对 象 的 语法 
new PDO ( 数据 源 名 称 户 名 ， 密码 ) 

















户 名 ”和 “密码 ”是 用 于 连接 MySQL 的 用 户 名 和 密码 ， 在 本 书 使 用 的 环境 中 ， 二 者 都 是 























“数据 源 名 称 ” 的 格式 如 下 所 示 。 
驱动 名 称 : host= 主机 名 ; dbname= 数据 库 名 
如 表 18-1 所 示 ， 驱 动 名 称 取 决 于 数据 库 的 种 类 。 





表 18-1 指定 驱动 名 称 的 字符 串 


数据 库 的 种 类 驱动 名 称 














MySQL mysql 
PostgreSOL pgsql 
Oracle G6 
SOL Server sqlsrv 




















“主机 名 ”中 记述 的 是 运行 MySQL 服务 器 的 主机 的 名 称 ， 在 本 书 使 用 的 环境 中 ， 该 主机 名 是 
“localhost”。 因 此 ， 如 果 在 本 书 使 用 的 环境 中 创建 用 于 操作 数据 库 dpl 的 PDo 对象， 就 需要 按照 如 
下 方式 进行 描述 。 

















new PDO("mysql:host=localhost;dbname=dbl1l","root","root"); 


将 创建 的 对 象 赋 给 变量 ， 这 种 值 为 对 象 的 变量 称 为 “对 象 变量 ”， 对 象 可 以 通过 该 变量 进行 操 
作 。 有 具体 示例 如 代码 清单 18-1 所 示 。 


代码 清单 18-1 connection.php 











<?php 
$s=new PDO("mysql:host=localhost;dbname=db1l", "root", "root"),; 
print " 连接 成 功 "; 


?> 




















执行 上 面 的 connection.php 会 创建 PDo 类 的 对 象 并 将 其 赋 给 对 象 变 量 ss。 我们 可 以 通过 变量 
$s 来 操作 MySQL 服务 器 。 如 果 连 接 成 功 ， 则 执行 第 2 行 代 码 ，Web 页 面 会 显示 “连接 成 功 ”。 

下 面 我 们 来 实际 连接 一 下 。 将 上 述 脚本 保存 在 发 布 的 文件 夹 中 ， 并 确认 MySQL 和 Apache 下 
在 运行 。 然 后 ， 在 浏览 器 中 输入 http://localhost/connection.php， 按 Enter 键 。 如 果 连 接 成 功 ， 则 会 
显示 图 18-2 的 画面 。 
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| 人 hapyvocalhosyconnecion 





连接 成 功 








18-2 ”执行 结果 ( 连接 成 功 时 的 情况 ) 

如 图 18-3 所 示 ， 如 果 显 示 出 “网 站 无 法 显示 该 页 面 ”等 信息 ， 我 们 可 以 从 服务 器 名 、 用 户 名 、 
密码 的 设置 是 否 有 问题 ,或 者 Apache 或 MySQL 本 身 是 否 正常 运行 等 方面 来 找 原因 。 页 面 的 布局 
和 信息 会 根据 浏览 器 的 不 同 而 发 生变 化 ， 一 般 都 会 显示 “500” 这 个 错误 。 该 错误 表示 当 输 出 动态 
页 面 时 程序 上 发 生 了 错误 。 















DE) htp://ocalhosyconnection.php 
x 


0@ 网 站 无 法 显示 该 页 面 


最 可 能 的 原因 是 : 
* 该 网 站 正在 进行 维护 。 
* 该 网 站 有 程序 错误 。 














你 可 以 尝试 以 下 操作 : 
@ ”刷新 该 页 面 。 
时 返回 到 上 一 页 。 


加 详细 信息 











18-3 ”执行 结果 ( 连接 失败 时 的 情况 ) 
总 之 ,我们 要 确保 可 以 成 功 地 连接 到 MySQL 服务 器 。 


18.2 使 用 PDO 执行 SQL 语句 


连接 到 MySQL 服务 器 之 后 ， 就 可 以 通过 PHP 脚本 执行 SQL 语句 了 。 我 们 向 前 面 出 现 多 次 的 数 
据 库 dbl 的 表 tbl 中 添加 记录 。 作 为 示例 ， 试 着 使 用 PHP 脚本 处 理 以 下 MySQL 监视 器 中 的 命令 。 

在 PDo 中 执行 SQL 语句 需要 使 用 query 方法 。 

query 方法 


ooosoooooooooooososeeooeo eeeoorossooosooooeososreoosoo eeesesoseoosoeoooososeossoooooooseoooeeooosoosoeooseoooeeosoeeooossooseooseooossosesosee 


: PDO 对 象 ->query (" 执行 的 SQL 语句 ") 


执行 方法 时 需要 给 对 象 变 量 加 上 符号 “- >”， 这 样 就 可 以 明确 执行 的 是 哪个 对 象 的 方法 了 。 这 
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次 我 们 将 new 创建 的 PDO 对 象 赋 给 了 对 象 变量 $s。 也 就 是 说 ， 当 通过 $s 向 数据 库 dbl 发 送 SQL 
语句 时 ， 代 人 码 需 要 编写 为 $s->query (...)。 


18.2.1 向 表 中 插入 记录 


一 一 一 一 一 一 一 一 一 一 一 一 





首先 在 表 tbl 中 添加 一 条 记录 。 表 tb1 的 当前 状态 和 要 添加 的 记录 如 下 所 示 。 






































> 表 tb1 的 数据 > 添加 到 表 tb1 中 的 记录 
empid name age empid name age 
A101 佐 茧 40 K777 吏 豆 太郎 20 
A102 高 桥 28 

A103 中 川 20 

A104 渡 边 23 

A105 西 泽 35 
































> 创建 并 执行 脚本 


试 着 创建 并 执行 PHP 脚本 。 请 创建 代码 清单 18-2 中 的 脚本 ， 将 该 脚本 命名 为 insert.php 并 保存 
到 发 布 的 文件 夹 中 。 


代码 清单 18-2 insert.php 





























<?php 

ScenewepPnol um So: a lo bname = dl Oo Oo 

$s->query ("INSERT INTO tbl VALUES('K777',' 吏 辟 太郎 ' ,20)"); 

用 于 操作 数据 库 dbl 的 PDo 对 象 的 创建 方法 和 18.1.3 节 相 同 〈 国 )。 

回 发 送 了 INSERT 语句 。 因 为 query 方法 的 参数 是 字符 串 ， 所 以 需要 使 用 " "或 ' :把 SQL 
语句 括 起 来 。 但 由 于 INSERT 语句 中 也 使 用 了 字符 串 ， 所 以 整个 INSERT 语句 用 " " 括 了 起 来 ， 








INSERT 语句 中 包含 的 字符 串 则 使 用 ' ' 括 了 起 来 。 这 部 分 内 容 会 在 后 面 进行 讲解 。 
保存 文件 后 ， 在 浏览 器 的 地 址 栏 中 输入 以 下 URL ， 然 后 按 Enter 键 。 





http://localhost/insert.php 


> 确认 成 功 添加 记录 





因为 这 次 的 PHP 脚本 仅 用 于 添加 记录 ， 所 以 结果 不 会 显示 在 浏览 咒 上 。 也 就 是 说 ， 我 们 无 法 











18.2 使 用 PDO 执行 SQL 语句 | 409 














在 浏览 器 上 确认 添加 记录 的 操作 是 成 功 了 还 是 失败 了 。 要 想 进行 确认 ， 就 需要 启动 MySQL 监视 器 








并 执行 SELECT 语句 。 下 面 就 来 实际 操作 一 下 。 








操作 方法 


加 





启动 MySQL 监视 器 。 
@ 输入 “use db1"。 
G@) 输入 “SELECT * FROM tbl;"o 












































执行 结果 

mysql> select * from tbl; 

Rn es SS 十 
empid name age 

于 车 ee 于 + 
ARA101 佐 芯 40 
R102 高 村 28 
RAR103 中 川 20 
RA104 渡 边 phe: 
Al05 西 泽 35 
K777 驶 豆 太 郎 20 

末 学 吉 汪 二 汪汪 每 td 虽 亲 台 汉 尝 学 避 十 

6 rows in set (0.00 sec) 


























执行 结果 中 如 果 显 示 了 新 添加 的 记录 ， 就 代表 成 功 了 。 





P 使 用 ' 或 "将 SQL 语句 括 起 来 
这 次 我 们 使 用 了 下 面 的 代码 发 送 SQL 语句 。 





$s->query ("INSERT INTO tbl VALUES('K777',' 吏 豆 太 郎 ',20)"); 














前 面 提 到 过 ， 因 为 使 用 query 方法 执行 的 SQL 语句 是 字符 串 ， 所 以 必须 用 “"” 或 “'” 将 其 

















括 起 来 。 但 是 ， 使 用 了 ' “的 SQL 语句 如 果 再 用 ，，' 括 起 来 ， 就 会 发 生 错误 。 例 如 执行 下 面 的 代 











码 就 会 发 生 错误 。 


$s->query ('INSERT INTO tbl VALUES('K777',' 跪 豆 太 郎 ',20)'); 





为 了 避免 发 生 错 误 ， 我 们 可 以 按 如 下 方式 编写 代码 。 


$s->query ('INSERT INTO tbl VALUES(\'K777\',\' 豌 豆 太 郎 \',20)'); 








在 内 部 “之 前 加 上 "\" 
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可 以 看 到 ， 上 面 的 代码 在 内 部 的 “' ”之 前 加 上 了 从” 进行 转 义 处 理 (一 5.3.2 节 ) 
当然 ， 也 可 以 像 下面 这 样 反 过 来 使 用 " "和 ' '。 









































$s->query ('INSERT INTO tbl VALUES("K777"," 跪 豆 太郎 ",20)'); 











本 书 统一 采用 了 示例 中 的 方法 来 编写 代码 。 








S 18.2.2 ”在 PHP 中 接收 SQL 语句 的 执行 结果 
通过 PDO 执行 SQL 语句 后 ， 我 们 并 没有 立刻 知道 结果 。 要 想 把 执行 结果 显示 在 浏览 器 上 ， 就 
需要 执行 SELECT 语句 并 对 其 结果 进行 处 理 。 
query 方法 也 可 以 用 来 执行 SELECT 语句 ， 如 果 执 行 了 用 于 返回 结果 的 SQL 语句 ，query 方 
法 就 会 返回 这 个 结果 。 下 面 的 程序 会 将 query 方法 返回 的 结果 赋 给 对 象 变量 sre。 
































$re=$s->query ("SELECT * FROM tb1"); 

















但 是 ， 即 使 执行 print $re,， 结果 也 不 会 显示 出 来 。 这 里 就 不 解释 具体 原因 了 ， 简 而 言 之 ， 
就 是 query 方 法 的 返回 值 不 是 单纯 的 字符 串 ， 而 是 一 个 对 象 (PDOStatement 对 象 )。 
PDOStatement 对 象 中 有 各 种 各 样 的 功能 ， 而 保存 SQL 语句 的 执行 结果 并 执行 后 续 处 理 正 是 它 的 
功能 之 一 。 

fetch() 方法 可 以 从 作为 SELECT 语句 结果 的 PDOStatement 对 象 中 取出 使 用 print 就 能 
输出 的 记录 。 

fetch 方法 


: PDOStatement 对 象 ->fetch () 
























































fetch() 方法 没有 参数 ， 返 回 值 是 记录 的 数组 。 因 为 要 对 quezy 方 法 的 结果 $re 执行 
fetch() 方法 ， 所 以 代码 要 写成 Sre->fetch()。 








$re=$s->query ("SELECT * FROM tb1"); 
$result=$re->fetch(); 





上 面 的 程序 对 query 方法 的 结果 ($re) 执行 了 fetch() 方 法， 并 将 其 赋 给 了 变量 
$result。s$result 是 一 个 数组 变量 (一 16.1.4 节 )。 第 1 行 各 列 的 值 分 别 是 $result [0]、 
$result [1] 、s$result [2]。 

最 终 会 取出 下 面 几 个 值 。 请 注意 ， 下 标 是 从 0 开始 的 ， 而 不 是 从 1 开始 的 。 图 18-4 是 相应 的 
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2 S$result[0] 、 列 empid “A101” 等 
2 $result[1] - 列 name “佐藤 ”等 
2 $result[2] ~ 列 age “40” 等 


SP 站 
| [| 
i$re 






lI” 
| 








A101 佐 芯 40 
2 


图 18-4 将 查询 结果 赋 给 数组 变量 









”fetch 方法 的 使 用 方法 


最 初 执行 fet ch 方法 时 ，SELECT 语句 的 结果 中 包含 的 第 1 条 记录 的 各 列 会 作为 数组 返回 ， 
第 2 次 执行 fet ch 方法 时 ,第 2 条 记录 的 各 列 会 作为 数组 返回 。 也 就 是 说 ， 只 要 像 下 面 这 样 根据 
记录 数 执行 fetch 方法 ， 就 可 以 取出 所 有 的 记录 。 
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但 是 ， 根 据 结 
在 很 多 情况 下 ， 我 1 
因此 ， 我 们 试 着 将 上 





























中 包含 的 记录 数 重复 执行 fet ch 方法 的 操作 方式 并 没有 什么 技术 含量 ,况且 
门 都 无 法 事先 知道 SELECT 语句 的 结果 中 会 包含 多 少 记录 。 
的 代码 改 为 下 面 这 种 形式 。 


























$re=$s->query ("SELECT * FROM tb1"); 


while($result=$re->fetch()){ 
print $result[0]; 


print $result[1]; 


print $result[2]; 











$re->fetch() 会 从 执行 SQL 语句 的 结果 中 返回 一 个 数据 数组 ， 如 果 没 有 行 可 读 取 就 会 返回 


FALSE。 也 就 是 说 ， 在 上 面 的 程序 
TRUE 则 执行 “print S$result ... 


> 添加 记录 后 立即 显示 结果 


此 外 ， 如 果 在 执行 SQL 语句 SELECT * FROM tbl 之 前 执行 INSERT INTO tbl 





VALUES () ， 就 可 以 立即 确认 添加 记录 后 的 结果 。 








Ph，fet cn 会 一 行 一 行 地 循环 读 取 ， 如 果 $re->fetch() 为 
”， 如 果 没 有 行 可 读 取 ， 则 返回 FALSE， 结 束 while 循环 。 




















下 面 是 添加 男 一 条 记录 ( 'K888' ，' 岗 村 花子 ，,25 ) 后 立即 使 用 SELECT 显示 数据 的 示例 。 
为 了 便于 阅读 ,我们 输入 了 换行 符 和 符号 “:”。 





代码 清单 18-3 insert_expression.php 





<?php 





oslols 
Ben 
SEE 
Ene 
en 
ETTE 





print " 连接 OK! <br>"; 
$s->query ("INSERT IN] 
STe=SS Souenmy (VOR 
while ($result=$re->fetch()){ 





sresult [0]; 
se 
Seesva an 
es 
Smesmle yl 


WU] oh SL 


$s=new PDO("mysql:host=localhost;dbname=db1l", "root", "root"),; 


rO tbl VALUES('K888',' 岗 村 花子 ' ,25)"); 
* FROM tb1"); 


因 目 








贺 用 于 将 SQL 语句 的 结果 赋 给 变量 $re， 回 用 于 将 全 部 结果 作为 数组 取出 并 显示 出 来 。 执 行 结 














如 图 18-5 所 示 。 





翁 httpy/localhostinsert_sxpression.php ”用 搜索 


励 localhost 
连接 OK! 
Al01- 估 藤 :40 
A102: 高 桥 :28 
3: 中 川 :20 
wk] 
A105: 西 泽 :35 
K777: 豌 豆 太 郎 :20 
K888: 岗 村 花子 :25 

















图 18-5 ”执行 结果 


18.3 ”异常 处 理 中 的 错误 处 理 
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在 因为 某 种 故障 而 无 法 连接 到 MySQL 数据 库 的 情况 下 ， 前 面 


将 介绍 数据 库 相 关 处 理 出 现 问题 时 的 应 对 策略 。 


D | 
介绍 的 程序 都 将 停止 处 理 。 本 闻 





18.1.3 节 介 绍 了 用 于 连接 到 MySQL 数据 库 的 程序 connection.php。 该 程序 在 发 生 错 误 的 情况 下 
会 显示 错误 提示 页 面 ， 但 不 会 显示 错误 发 生 的 原因 。 这 就 给 我 们 带 来 了 困扰 。 因 此 ， 我 们 试 着 让 程 





序 显示 出 错误 发 生 的 原因 。 
在 PHP 中 ， 执 行 过 程 中 发 生 的 问题 ( 错误 ) 称 为 





已 六 已 
天 用 。 天 











常 处 理 是 指 在 发 生 异 常 的 情况 下 执 


行 的 处 理 。 当 发 生 异 常 时， 程序 会 根据 错误 种 类 自动 生成 各 种 各 样 的 对 象 ， 所 以 我 们 可 以 使 用 该 对 


象 来 编写 异常 处 理 。 
执行 异常 处 理 需 要 用 到 “try ~ catch” 语 句 。 


try ~ catch 





生 异 常 的 处 理 
名 称 接收 异常 的 变量 ) { 
在 发 生 异 常 的 情况 下 执行 的 处 理 


























代码 清单 18-4 在 18.1.3 节 介 绍 的 程序 connection.php 
行 异 常 处 理 。 





! 添 加 了 异常 处 理 。 密 码 出 现 错误 ， 因 此 
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代码 清单 18-4 try_catch.php 





<?php 

try{ 
$s=new PDO("mysql:host=localhost;dbname=dbl", "root", "nisepass"); 
print " 连接 成 功 "; 


























}catch (PDOException $e){ 
print "下 面 是 错误 的 内 容 : " .$e->getMessage () ; 

} 

pw 














在 连接 数据 库 失 败 的 情况 下 会 引发 名 为 PDOException 的 异常 ， 因 此 使 用 了 变量 $e 接收 
PDOException ( 国 )。 有 具体 内 容 就 不 详细 介绍 了 ， 大 家 只 要 知道 PDOException 对 象 中 定义 了 各 
种 各 样 的 方法 ， 其 中 getMessage 方法 可 以 用 来 获取 错误 信息 即 可 。 


getMessage 方法 

















回 会 将 字符 串 “ 下 面 是 错误 的 内 容 :” 和 用 se->getMessage () 取得 的 错误 信息 结合 起 来 
显示 。 
代码 清单 18-4 的 程序 会 在 连接 数据 库 失 败 的 情况 下 显示 图 18-6 的 内 容 。 

















x 
下 面 是 错误 的 内 容 ，SQLSTATE[HY000] [1045] Access denied for user 'root'@'localhost (using ne 可 以 得 知 连接 数 据 
YES) 


库 的 要 求 被 拒绝 了 














图 18-6 ”执行 结果 


C 18.3.1 SQL 的 错误 处 理 

正如 前 面 的 示例 所 展示 的 那样 ， 当 PHP 与 MySQL 连接 失败 时 会 发 生 异 常 。 但 是 ， 连 接 到 
MySQL 时 生成 的 PDO 对 象 默 认 设 置 为 连接 后 即使 发 生 错 误 也 不 进行 异常 处 理 。 也 就 是 说 ， 如 果 想 
执行 MySQL 连接 失败 以 外 的 异常 处 理 ， 就 需要 修改 初始 设置 ( 属性 )。 














为 此 ， 我 们 需要 使 用 setAt 
setAttribute 方法 


错误 相关 的 属性 通过 值 PDo : 
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tribute 为 法 8 











:ATTR _ERRMODE 来 表示 。 因 此 , 将 属性 设置 为 PDO: :ERRMODE_ 








EXCEPTION， 发 生 错 误 时 就 会 生成 PDOException 对 象 。 





$s->setAttribute (PDO: :ATTR ERRMODE,PDO::ERRMODE EXCEPTION); 


在 执行 SQL 语句 的 过 程 中 发 生 异 常 就 会 显示 错误 内 容 的 程序 如 代码 清单 18-5 所 示 。 该 程序 在 
18.3 节 的 程序 insert_ expression.php 中 添加 了 显示 错误 内 容 的 处 理 。 


代码 清单 18-5 exception.php 

































































<?php 
try{ 
$s=new PDO("mysql:host=localhost;dbname=db1l", "root", "root"); 
$s->setAttribute (PDO: :ATTR ERRMODE,PDO: : ERRMODE EXCEPTION); 
BELntE." 连接 OK! <br>"; 
$s->query ('INSERT INTO tbl VALUES ("K888"," 岗 村 花子 ",555)'); 
Sre=$s->query ("SELECT * FROM tb bad"); 
while ($result=$re->fetch()){ 
print $result[0]; 
eeuliol ee Wd LL 
Biltnt Se ee 
olson Ws 
print $result [2] ; 
alebale Wealone 
} 
}catch (PDOException $e){ 
print " 下 面 是 错误 的 内 容 : " .$e->getMessage(); 
} 
?> 
生成 PDO 对 象 后 ， 通 过 setAttribute 方法 更 改 错误 模式 ， 以 便 在 执行 SQL 时 能 进行 异常 
处 理 ( 国 )。 
回 对 不 存在 的 表 执 行 了 SELECT 语句 ， 所 以 这 里 会 发 生 异 常 (PDOException 对 象 )。 
回 用 于 将 异常 内 容 显示 在 画面 上 。 
执行 结果 如 图 18-7 所 示 。 
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cls 
SE 天 
E 
连接 OK! 

下 面 是 错误 的 内 容 ，SQLSTATE[42S02]: Base table or view not found: 1146 Table 'dbl tb_bad' doesn't 
exist 








可 以 看 到 表 名 错 了 








图 18-7 执行 结果 
另外， 为 了 简化 代码 ， 本 书 介 绍 的 示例 中 没有 编写 错误 处 理 。 如 果 不 知道 哪里 发 生 了 错误 ,请 
试 着 使 用 try ~ catch 语句 ， 并 使 用 setAttripute 方法 修改 错误 处 理 模 式 。 


显示 数据 库 中 存在 的 表 


要 想 显示 数据 库 中 存在 的 表 ， 就 需要 使 用 query 方法 发 送 MySQL 的 SHOW TABLES 命令 (一 4.5.1 节 ) 
执行 查询 。 


下 面 的 脚本 用 于 输出 数据 库 db1 中 存在 的 表 。 执 行 结果 如 图 18-8 所 示 。 



































PP show_tb.php 


<?php 
$s=new pdo ("mysql:host=localhost;dbname=db1l", "root", "root"); 
$re=$s->query ("SHOW TABLES"); 
: while ($result=$re->fetch()){ 
: print $result [0] ; 
Beint. "<br 


¢ | Er -0 co 





18-8 执行 结果 
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18.4 总 结 


本 章 介绍 了 以 下 内 容 。 


@ 使 用 PHP 连接 MySQL 数据 库 的 方法 
@ 使 用 PHP 执行 查询 的 方法 

@ 使 用 PHP 添加 记录 的 方法 

@ 使 用 PHP 显示 查询 结果 的 方法 


> 自我 检查 
下 面 检查 一 下 本 章 学 习 的 内 容 是 否 全 部 理解 并 掌握 了 。 


口 能 够 使 用 Ppo 连接 数据 库 
口 能 够 使 用 query 方法 执行 查询 
口 能 够 使 用 fetch 方法 显示 查询 结果 























i 练习 题 
问题 1 
创建 PHP 脚本 ， 在 表 tb 中 以 列 sales 的 值 大 于 等 于 50 的 记录 为 对 象 ， 按 照 列 empid 进行 分 组 ， 并 降序 
































显示 各 组 销售 额 平均 值 大 于 等 于 120 万 元 的 记录 。 执 行 的 SQL 语句 要 使 用 8.8 节 参 考 答案 中 的 内 容 ， 即 下 面 
的 代码 。 





























: SELECT empid,AVG (sales) 
FROM tb 
C WHERE sales>=50 
GROUP BY empid 
; HAVING AVG(sales) >=120 
ORDER BY AVG(sales) DESC; 





员工 号 ，A101 平均 销售 额 ，242.0000 
员工 号 ,A104 平均 销售 额 ，137.0000 
员工 号 A102 平均 销售 额 ，129.5000 





HINT 
使 用 Here Document (一 17.7.1 节 ) 设置 SOL 语句 操作 起 来 会 比较 轻松 。 
【2 
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局 参考 答案 














创建 代码 清单 18-6。 


代码 清单 18-6 empid_avg.php 





<?php 
$s=new PDO("mysql:host=localhost;dbname=db1l", "root", "root"),; 


$q=<<<eot 
SELECT empid,AVG (sales) 
FROM tb 
WHERE sales>=50 
GROUP BY empid 
HAVING AVG (sales) >=120 
ORDER BY AVG(sales) DESC; 


ect ; 


Sre=$s->query ($9q); 
while ($result=$re->fetch()){ 





print " 员工 号 : "; 
print $result[0]; 
print " 平均 销售 额 : "; 
print $result[1]; 





DELNt Wb 











第 5 部 分 
MySQL + PHP 实 践 


使 用 PHP 和 MySQL 制 作 一 个 简易 公告 板 
发 布 到 互联 网 上 时 需要 注意 的 地 方 


创建 一 个 实用 公告 板 























使 用 PHP 访问 MySQL 之 后 ， 试 着 制作 一 个 简单 的 Web 应 用 程序 连接 到 MySOL。 
在 第 5 部 分 的 内 容 中 ， 我 们 将 学 习 如 何 使 用 PHP 和 MySQL 制作 公告 板 ， 以 及 发 布 到 互 
联网 上 时 需要 注意 哪些 地 方 等 。 



































第 19 章 ”使 用 PHP 和 MySOL 制 
作 一 个 简易 公告 板 


终于 要 使 用 MySQL 和 PHP 制作 一 个 Web 应 用 程序 了 。 我 
们 先 来 复习 一 下 前 面 的 知识 ， 然 后 试 着 制作 一 个 简易 的 公告 板 。 




































































19.1 制作 一 个 简易 公告 板 


19.1.1 简易 公告 板 的 结构 


我 们 来 制作 一 个 简易 的 公告 板 ， 顺 便 整 理 一 下 前 面 学 习 的 MySQL 和 PHP 的 知识 。 简 易 公告 
板 ， 顾 名 思 义 ， 不 是 一 个 复杂 的 Web 应 用 程序 。 这 个 公告 板 只 会 连接 到 数据 库 执行 显示 记录 、 写 
入 姓名 和 消息 、 删 除 记录 、 查 询 记 录 这 4 个 简单 的 处 理 ( 见 图 19-1 )。 

我 们 要 制作 的 公告 板 没有 个 人 身份 验证 ， 并 且 可 以 自由 地 进行 删除 ， 所 以 如 果 投 入 实际 应 用 ， 
在 安全 方面 还 存在 隐患 ， 不 过 在 本 地 服务 器 中 使 用 是 完全 没有 问题 的 。 
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es imple.html ”用 搜索- 办 ~ 
愿 localhost 
显示 消息 
显示 消息 


输入 姓名 
输入 消息 


























发 送 消息 
输入 要 删除 的 编号 | 
发 闫 删除 编号 





























输入 查询 的 关键 字 | 
音 询 




















19-1 简易 公告 板 的 首页 ( simple.html ) 


为 了 让 MySQL 和 PHP 的 运行 过 程 更 易于 理解 ， 我 尽力 缩减 了 标签 的 种 类 。 如 果 有 可 以 使 用 的 HTML 
标签 ， 使 用 print 输出 即 可 。 大 家 可 以 按照 自己 的 喜好 进行 设计 。 图 19-2 是 显示 所 有 消息 的 画面 。 




















和 3S Se jsimple_select php ofa 有 


1 : 西 泽 梦 路 : 终于 完成 了 。 请 大 家 尽情 地 进行 输入 吧 ! 
2 中 川 : 大 家 好 ， 我 是 中 川 。20 岁 。 是 D 有 限 公司 总 务 部 的 。 请 多 多 关照 。 
3 : 佐 蔽 : 成 庆生 庆 有 点 儿 告 间 * 我 是 年 龄 最 大 的 佐 芒 。 和 拜托 了 





返回 首页 





19-2 显示 所 有 消息 的 画面 ( simple_select.php ) 


Ge 19.1.2 ”创建 数据 库 和 表 


首先 创建 在 这 个 Web 应 用 程序 中 进行 操作 的 表 。 在 数据 库 dbl 中 创建 列 结构 如 表 19-1 所 示 的 
表 tbk。 





表 19-1 表 tbk ( 简易 公告 板 消息 表 ) 


列 名 empid name mess 
INT 
定义 AUTO_INCREMENT VARCHAR (100) VARCHAR (100) 





PRIMARY KEY 
列 的 作 保存 消息 的 连续 编号 保存 写 入 的 姓名 保存 消息 
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构成 表 tbk 的 列 empid、 列 name 和 列 mess 分 别 用 于 保存 编号 、 姓 名 和 消息 。 其 中 ,， 列 empid 
有 具有 自动 连续 编号 功能 (一 6.8 节 )。 
大 家 应 该 还 记得 表 的 创建 方法 吧 (一 4.4.3 节 ),。 请 事先 启动 MySQL 监视 器 ， 创建 表 tbk。 



































19.1.3 简易 公告 板 的 文件 结构 


这 个 简易 的 公告 板 具 有 对 记录 进行 显示 、 插 入 、 删 除 和 查询 的 功能 。 我 们 来 创建 用 于 实现 各 个 
功能 的 PHP 脚本 吧 。 


























> 本章 中 创建 的 示例 
我 们 将 创建 以 下 PHP 脚本 文件 。 
[cy 





一 一 simple_select.php …… 显示 记录 
simple_insert.php 7..... 插入 记录 


| simple_delete.php ....... 删除 记录 
simple_search.php.… 查询 记录 
simple.html -nsseeeeees 行 上 面 的 4 个 脚本 


我 们 来 看 一 下 首页 和 各 个 脚本 文件 之 间 的 关系 ( 见 图 19-3 )。 






































simple.html 




















~ | S| Sq 
图 19-3 简易 公告 板 的 结构 


首页 上 有 4 个 表单 ， 每 个 表单 都 可 以 执行 各 自 的 脚本 ， 这 些 脚 本 分 别 具 有 显示 记录 、 插 人 记 
录 、 删 除 记录 和 查询 记录 的 功能 。 

例如 ， 当 用 户 执行 插入 记录 的 操作 时 ， 在 文本 框 内 输入 “姓名 ”和 “消息 ”， 点 击 “ 发 送 消息 ” 
按钮 。 于 是 ， 数 据 就 会 发 送 到 具有 插入 记录 功能 的 simple_insert.php 中 。 
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19.2 创建 首页 


我 们 先 来 创建 首页 ( simple.html )。 





19.2.1 调用 消息 显示 脚本 的 表单 

第 一 个 要 调用 的 是 具有 显示 功能 的 脚本 。 创 建 一 个 表单 ， 该 表单 用 于 调用 显示 表 tbk 中 所 有 记 
录 的 脚本 simple_select.php。 

在 这 个 简易 公告 板 中 ， 发 送 数据 的 方式 均 为 POST。 因 此， methodq="POST"。 在 这 个 表单 中 
设置 一 个 具有 提交 ( submit ) 功能 的 按钮 (一 17.8.2 节 ), 通过 value=" 显示 消息 " 将 按钮 上 显 
示 的 字符 串 设 置 为 “显示 消息 ”。 

调用 显示 功能 的 表单 内 容 如 下 所 示 。 图 19-4 是 相应 的 示意 图 。 

































































<form method="POST" action="simple select.php"> 
<div> 显 示 消 息 </div> 

<input type="submit" value=" 显 示 消 息 "> 

</form> 









<form method="POST"… 


显示 消息 








</form> 

















| A | 


simple_select.php 


图 19-4 显示 功能 


























19.2.2 ”调用 插入 脚本 的 表单 


接 下 来 要 调用 的 是 具有 插入 功能 的 脚本 。 创 建 一 个 用 于 调用 脚本 simple_insert.php 的 表单 。 针 
对 <form> 标签 ， 只 修改 相应 PHP 文件 的 指定 部 分 ， 将 该 部 分 修改 为 <form method="POST" 


action="simple insert .php">o 





424 | 第 19 章 使 用 PHP 和 MySQL 制作 一 个 简易 公告 板 


创建 用 于 输入 姓名 和 消息 的 文本 框 ， 并 使 用 name 属性 设置 用 于 识别 数据 的 元 素 名 称 
(一 17.8.2 节 )。 将 姓名 文本 框 和 消息 文本 框 的 name 属性 分 别 设置 为 al 和 a2。 
调用 插入 功能 的 表单 如 下 所 示 。 图 19-5 是 相应 的 示意 图 。 





对 用 于 输入 信息 的 文本 框 设 置 了 size 属性 来 规定 其 大 小 。 








19-5 插入 功能 


19.2.3 ”调用 删除 脚本 的 表单 


下 面 要 调用 的 是 具有 删除 功能 的 脚本 。 创 建 一 个 用 于 调用 脚本 simple_delete.php 的 表单 ， 文 本 
框 中 输入 的 是 要 删除 的 消息 编号 ， 将 该 文本 框 的 name 属性 设置 为 bl。 图 19-6 是 相应 的 示意 图 。 











中 size 属性 定义 的 是 可 见 的 字符 数 ， 也 就 是 在 这 个 输入 框 中 最 多 能 看 到 的 字符 数 ， 其 余 的 字符 在 前 后 隐藏 。 一 一 译 者 注 
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<form method="POST"…> EE 2 
要 删除 的 编号 


发 送 要 删除 的 编号 ke 
</form> 
<input type="submit"…: | 
































simple_delete.php 


删除 























图 19-6 ”删除 功能 


19.2.4 调用 查询 脚本 的 表单 


最 后 要 调用 的 是 具有 查询 功能 的 脚本 。 创 建 一 个 用 于 调用 脚本 simple_search.php 的 表单 。 文 本 
框 中 输入 的 是 要 查询 的 关键 字 ， 查 询 消息 中 包含 的 字符 串 。 将 该 文本 框 的 name 属性 设置 为 cl。 
图 19-7 是 相应 的 示意 图 。 





















































<form method="POST" action="simple search.php"> 
<div> 输 入 查询 的 关键 字 <input type="text" name="cl"></div> 
<input type="submit" value=" 查 询 "> 

</form> 








<form method="POST"...> 
查询 的 关键 字 


</form> ES 
<input type="submit"... 


simple_search.ph 
| 


图 19-7 ”查询 功能 


POST 发 送 


忌 古 
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19.2.5 创建 首页 


一 一 一 一 一 一 一 


下 面 来 汇总 一 下 前 面 的 内 容 ， 创 建 首 页 simple.html。Web 页 面 simple.html 可 以 调用 分 别 拥有 











项 





示 、 插 和 人 入、 删除 和 查询 功能 的 PHP 脚本。 请 创建 该 Web 页 面 ， 并 将 划 
C:\MAMP\htdocs ) 中 。 





代码 清单 19-1 simple.html 





保存 在 发 布 的 文件 夹 





<!DOCTYPE html> 

<html> 

<head> 

<meta charset="UTF-8"> 
</head> 

<body> 


<p> 

<form method="POST" action="simple select.php"> 
<div> 显示 消息 </div> 

<input type="submit" value=" 显示 消息 "> 

</form> 

</p> 


<p> 

<form method="POST" action="simple insert.php"> 

<div> 

输入 姓名 <input type="text" name="al"> 

</div> 

<div> 输入 消息 <input type="text" name="a2" size=150></div> 
<input type="submit" value=" 发 送 消息 "> 

</form> 

/pS 


<p> 

<form method="POST" action="simple delete.php"> 

<div> 输入 要 删除 的 编号 <input type="text" name="bl"></div> 
<input type="submit" value=" 发 送 要 删除 的 编号 "> 

</form> 

</p> 


<p> 

<form method="POST" action="simple search.php"> 

<div> 输入 查询 的 关键 字 <input type="text" name="c1l"></div> 
<input type="submit" value=" 查询 "> 

</form> 

</p> 


</body> 
</html> 
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19.3 ”创建 分 别 具 有 显示 、 插 入 、 删 除 和 查询 功能 的 PHP 脚本 


首页 已 经 创建 守成。 下面 我 们 来 创建 分 别 具 有 显示 、 插 入、 删除 、 查 询 功能 的 PHP 脚本 。 




















从 首页 调用 的 PHP 文件 中 都 包含 “连接 到 数据 库 ” 和 “显示 当前 所 有 记录 ”的 处 理 。 这 两 
个 处 理 的 代码 具体 如 下 ， 其 内 容 已 经 在 第 18 章 介绍 过 了 。 理 解 了 这 一 点 ， 程 序 编写 起 来 就 会 更 加 
容易 。 


























$s=new PDO("mysql:host=localhost;dbname=dbl","root","root"); 
Sre=$s->query ("SELECT * FROM tbk ORDER BY empid"); 
while($result=$re->fetch()){ 

print $result[0]; 

Peint on 

print Sresult hl 

BELnt 0 

pelint Sresultelb2l; 

print Mbre > 


局 19.3.2 用 于 显示 记录 的 PHP 肢 本 


首先 来 思考 一 下 如 何 编写 一 个 只 要 点 击 按钮 就 会 显示 表 tbk 全 部 记录 的 PHP 脚本 ( 图 19-8 )。 

如 果 仅 从 SQL 语句 的 方面 考虑 ， 只 要 执行 SELECT * FROM tbk 即 可 ， 但 这 样 一 来 ， 我 们 便 
无 法 得 知 记录 的 显示 顺序 了 。 因 此 ， 为 了 让 记录 按照 列 empid 的 顺序 显示 ， 脚 本 中 要 加 上 ORDER 
BY empid (一 8.5.1 节 )。 执 行 SQL 语句 和 显示 结果 的 脚本 与 18.3 节 相 同 。 






































$s=new PDO ("mysdl :host=1localhostj;dqpbname=dqb1" "root","root"); 
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列 pa 的 顺序 
8 @ 


Sre=$s->query ("SELECT * FROM tbk ORDER BY empid"); 
~ 




















Se ea ee 
print $result [0] ; 
BrinE YT 2 
print $result [1]; 
print " : "; @------------------- 险 出 数组 的 值 
print $result [2]; 








DELNt, <bESY, 





19-8 ”simple_select.php 的 结构 
整理 一 下 上 面 的 内 容 ， 脚 本 就 会 如 代码 清单 19-2 所 示 。 


代码 清单 19-2 simple_select.php 








<?php 
ini set ("display errors", On); 


$s=new PDO("mysql:host=localhost;dbname=db1l", "root", "root"),; 


$re=$s->query ("SELECT * FROM tbk ORDER BY empid"); 
while ($result=$re->fetch()){ 

print $result [0] ; 

这 下 下 

print Sresult [ll] 

BEL We Ws 

print Srestilt [2] 3 

BELnt. VebeES™, 


} 











口 





贝 </a>"; 





print "<br><a href='simple.html'> 返 














?> 














创建 好 名 为 simple_select.php 的 文件 后 ， 将 其 保存 到 发 布 的 文件 夹 中 。 


Ge 19.3.3 ”用 于 插入 记录 的 PHP 脚本 


下 面 是 用 于 插入 姓名 和 消息 的 脚本 ( 见 图 19-9 )。 
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在 首页 的 name="al" 和 name="a2" 的 文本 框 中 分 别 输入 姓名 和 消息 。 

具有 搬入 功能 的 脚本 simple insert.php 接收 了 这 两 个 数据 ， 并 将 它们 分 别 插入 列 name 和 列 mess 中 。 

来 自发 送 方 simple.html 的 数据 可 以 通过 $s_POST ["al"] 和 8$ PosT ["a2"] 接收 (一 17.8.3 
节 )。 出 现 多 次 "” "和 ' 会 使 代码 变 得 复杂 ， 所 以 我 们 将 接收 的 两 个 数据 $_PosT [ "al" ] 和 
$_POST [ "a2" ] 分别 赋 给 变量 Sal_ da 和 $a2_q。 














$s=new PDO ("mysql:host=localhost;dbname=db1l", "root","root"); 








name="al" 一 一 姓名 




















$al d=$ POST["al"]; 


$a2_d=$_POST["a2"]; 
9 
























发 送 INSERT 语 名 ~ 












$s->query ("INSERT INTO tbk 





(name,mess) VALUES ('$al d','$a2 qd')"); 






























































name mess 
a eT 
一 二 -一 i 
@ @ 






Sre=$s->query("SELECT * FROM tbk ORDER BY bang" ) ; 









while(S$result=Sre->fetch())1{ 
print $result [0] ; 
Ein 








print Sresult [1] ; 偷 出 数组 的 值 


9 
1 
1 
1 
1 
1 
1 
1 
1 
1 
1 
1 
1 
1 
1 

过 











oh he 全 于 
print S$result [2] ; 


hi 二 hs 








19-9 simple_insert.php 的 结构 
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下 面 是 








LU 


于 将 接收 的 数据 插入 到 表 tbk 的 列 name 和 列 mess 中 的 SQL 语句 。 











INSERT INTO tbk (name,mess) VALUES ('$al d','$a2 d') 
发 送 此 SQL 语句 的 代码 如 下 所 示 。 


Said=sepOSm[l ral; 
Sa2Bq=S pOSm la 
$s->query ("INSERT INTO tbk (name,mess) VALUES ('$al d','$a2 d')"); 


请 注意 ，SQL 语句 INSERT INTOx xVALUES ('$al d','$a2 d') 用 " 

















(~—*162.3.) 


那么 , 在 $al_a 上 添加 ， ' 就 没有 问题 吗 ” 相 信 有 不 少 读者 都 有 这 样 的 担心 。 








" 括 了 起 来 。 如 





使 用 " "将 SQL 语句 括 起 来 ， 其 中 的 变量 $al da 和 $a2_da 就 会 被 解析 为 具体 的 字符 串 


的 确 , 用 ' ， 


括 起 来 的 变量 会 作为 字符 串 来 处 理 , 但 是 请 放心 ， 在 使 用 " " 的 情况 下 ，' “' 会 被 当成 单纯 的 字 























符 。 所 以 ， 如 果 将 “ 西 泽 ” 赋 给 变量 $a_d， 代 码 就 会 变 成 下 面 这 样 。 











INSERT INTO ... VALUES (' 西 泽 '...) 








在 上 述 内 容 的 基础 上 添加 连接 数据 库 、 显 示 记 录 的 处 理 ， 以 及 指向 首页 的 链接 
码 清 单 19-3 所 示 。 





代码 清单 19-3 simple_insert.php 


， 脚 本 就 会 如 代 





<?php 
$s=new PDO("mysql:host=localhost;dbname=db1", "root", "root"); 
$al d=$ POST["al"]; 
$a2 d=$ POST["a2"]; 
$s->query ("INSERT INTO tbk (name,mess) VALUES ('$al d','$a2 d')"); 
$Sre=$s->query ("SELECT * FROM tbk ORDER BY empid"); 
while($result=$re->fetch()){ 
print $result[0]; 








DELnt VY i Wy 

Bbrint Sresult [ll] 

oh An ha oe ee 
print $result [2] ; 
BELNE ohae 

} 


print "<br><a href='simple.html'> 返回 首页 </a>"; 





























人 > 














创建 好 名 为 simple_insert.php 的 文件 后 ， 将 其 保存 到 发 布 的 文件 夹 中 。 
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19.3.4 ”用 于 删除 记录 的 PHP 脚本 


下 面 来 看 用 于 删除 记录 的 脚本 ( 见 图 19-10 )。 

simple delete.php 用 于 接收 输入 到 文本 框 name="b1" 中 的 数据 ， 并 删除 与 表 tbk 的 列 empid 相 
匹配 的 记录 。 

来 自发 送 方 simple.html 的 数据 可 以 通过 $_POST [ "bl1" ] 接收 。 我们 把 $_POST [ "bi" |] 的 
值 赋 给 变量 $b1l_q。 














name="b1" 连接 
删除 的 编号 











EN 


Sbl d=$ POST["bl"]; 
_ 


_ 








es v r--- 相 匹配 的 empia 


1 
empid=$b1 qd"|); 





















$s->query ("DELETE FROM tbk WHERE 











使 用 SELECT 显示 


19-10 simple_delete.php 的 结构 
SQL 语句 DELETE FROM tbk WHERE empid=$bl qd 用 于 删除 列 empid 为 sbl da 的 记录 。 
因此 ， 用 于 发 送 此 SQL 语句 的 代码 如 下 所 示 。 














SblEaq=sPOSERIPE 
$s->query ("DELETE FROM tbk WHERE empid=$b1 d"); 


将 共同 的 处 理 内 容 添加 进来 ， 整 理 一 下 代码 。 于 是 ， 脚 本 就 如 代码 清单 19-4 所 示 。 该 脚本 具 
有 删除 记录 并 将 删除 后 所 有 记录 显示 出 来 的 功能 。 


代码 清单 19-4 simple_delete.php 





<?php 
$s=new PDO("mysql:host=localhost;dbname=db1l", "root", "root"); 


$b1 d=$ POSTI["b1"]; 
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$s->query ("DELETE FROM tbk WHERE empid=$bl1 d"); 
$re=$s->query ("SELECT * FROM tbk ORDER BY empid"); 
while ($result=$re->fetch()){ 

print $result [0] ; 

DEENt WW 

print $result [1]; 

Beint WY 于 

print $result [2] ; 

DEINnt. Veber 


} 

















print "<br><a href='simple.html'> 返回 首页 </a>"; 














昌吉 











创建 好 名 为 simple_delete.php 的 文件 后 ， 将 其 保存 到 发 布 的 文件 夹 中 。 


19.3.5 ”用 于 查询 记录 的 PHP 脚本 


最 后 一 个 脚本 用 于 提取 列 mess 中 含有 指定 字符 串 的 记录 (图 19-11 )。 
simple_search.php 用 于 接收 输入 到 文本 框 name="cl1" 中 的 数据 ， 并 提取 列 mess 中 含有 该 字符 
串 的 记录 。 


























| 连接 


name="cl" 查询 关键 字 



























































人 
~ 
ee v 查询 列 mess 中 含有 该 字 
“sy Ar cb *: 
l r= qd SELECT SS 符 串 的 记录 
发 送 SELECT 语 名 ww 

_ 1 

f Er 

© ”~ 全 [5 

$re=$s->query ("SELECT * FROM tbk WHERE mess LIKE '%$c1 d%'"); 












| 使 用 SELECT 显示 


图 19-11 simple_search.php 的 结构 
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我 们 使 用 $_POST [ "cl" ] 接收 来 自发 送 方 simple.html 的 数据 ， 并 将 此 值 赋 给 变量 sc1_ ad。 

对 于 列 mess 中 包含 “$c1l_aq 的 值 ” 的 记录 ， 用 于 提取 该 记录 的 SQL 语句 稍微 有 些 复杂 。 条 件 
“ 列 mess 的 数据 中 包含 x x ”表示 “x x 的 前 面 和 后 面 可 以 是 任何 内 容 "， 所 以 语句 要 写 为 mess 
LIKE '%x x%' (一 8.3.4 节 ) 因此， 当 提 取 列 mess 中 包含 $cl_a 的 值 的 记录 时 ， 需 要 使 用 如 
下 SQL 语句 。 
































SELECT * FROM tbk WHERE mess LIKE '%$c1 dS%!' 











另外 ， 为 了 能 显示 查询 结果 ， 我 们 使 用 变量 来 接收 通过 query 方法 发 送 SQL 语句 的 结果 。 这 
部 分 脚本 如 下 所 示 。 








$c1l d=$ POSTI["c1"]; 
$re=$s->query ("SELECT * FROM tbk WHERE mess LIKE '%$c1 d%'"); 





整理 一 下 前 面 的 内 容 ， 然 后 添加 共同 的 处 理 部 分 。 于 是 ， 脚 本 就 如 代码 清单 19-5 所 示 。 该 脚 
本 具有 查询 记录 并 将 符合 条 件 的 记录 显示 出 来 的 功能 。 


代码 清单 19-5 simple_search.php 











<?php 
$s=new PDO("mysql:host=localhost;dbname=db1l", "root", "root"),; 


$c1 d=$ POST["c1"]; 


$re=$s->query ("SELECT * FROM tbk WHERE mess LIKE '%$c1 d%'"); 
while ($result=$re->fetch()){ 

print $result [0]; 

Belnt VW 

print $result [1] 7; 

lsh oy ous LE 

print $result [2]; 

DELntE VBE 


} 

















brint “<br><a href='simple.html'> 返 可 自由 </a>"; 














> 














创建 好 名 为 simple_search.php 的 文件 后 ， 将 其 保存 到 发 布 的 文件 夹 中 





[e] 


将 simple.html、simple select.php、simple insert.php、simple delete.php 和 simple_ search.php 全 
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部 保存 到 发 布 的 文件 夹 中 后 ， 确 认 MySQL 和 Apache 的 运 
在 浏览 器 的 地 址 栏 中 输入 http://localhost/simple. ny ee 19.1.1 节 介 绍 


的 首页 。 


旦 下 在 公告 板 上 输入 标签 的 技巧 


公告 板 有 时 候 也 会 放 在 网 上 使 用 ， 我 们 下 面 要 讨论 的 就 是 这 个 环 手 的 问题 。 输 入 带 标签 的 消息 之 后 
告 板 也 会 把 该 标签 反映 出 来 。 本 






































BB 介绍 的 简易 公告 板 就 是 这 类 公告 板 中 的 一 个 。 











结果 ， 并 使 用 这 个 简易 公告 板 。 
如 的 简易 公告 板 


， 有 些 公 


我 们 来 看 一 个 例子 。 请 在 simple.html 的 输入 消息 一 栏 中 输入 并 发 送 “<font size="7"color="red"> 


遇 到 麻烦 了 !</font>” 














之 外 的 知 洒 


如 果 该 机 制 只 
参考 20.4.3 节 。 


















































































































































a 
OG venocahossimpie tm -0 他 关切 














输入 姓名 | 危险 的 人 
输入 消息 
<font size="7"color="red"> 罗 到 麻烦 了 ! </font> 

发 送 消 息 
输入 要 删除 的 编号 带 标签 发 送 消息 

发 送 删 除 编号 
























































输入 查询 的 关键 字 | 
音 询 

















19-12 ” 带 标 签发 送 消息 





: 西 泽 梦 路 : re 请 大 家 尽情 地 进行 输入 吧 ! 
: 中 川 : 大 家 好 ， 我 是 中 川 。20 岁 。 是 D 有 限 公司 总 务 部 的 。 请 多 多 关照。 
: 佐 节 :感觉 稍微 有 点 儿 老 啊 。 我 是 年 龄 最 大 的 佐 谋 。 拜 托 了 


4: 危险 的 人 : 好 遇 到 麻烦 了 ! 


返回 首页 





图 19-13 ”执行 结果 








这 种 带 有 <font> 标签 的 消息 ( 图 19-12 )。<font> 标签 用 于 设置 文字 的 尺寸 和 颜色 ， 
虽然 它 不 能 在 HTML 5 中 使 用 ， 但 是 在 大 多 数 浏览 器 中 该 标签 是 可 以 正常 工作 的 ( 图 19-13 )。 
如 果 仅 仅 用 来 修饰 文本 ， 倒 也 没有 什么 危害 ， 但 如 果 标 签 能 够 插入 进去 ， 就 意味 着 可 以 使 用 脚本 运行 








出 预想 


是 单纯 输出 字符 串 ， 那 么 发 送 的 所 有 标签 也 会 输出 出 来 。 关 于 删除 标签 的 相关 内 容 ， 大 家 可 以 








19.4 总 结 | 435 


19.4 总 结 


本 章 介绍 了 以 下 内 容 。 


@ 如 何 使 用 PHP 和 MySQL 创建 Web 应 用 程序 
e 针 对 分 别 具 有 显示 、 插 入 、 删 除 、 查 询 功能 的 脚本 ， 如 何 来 创建 调用 这 些 脚本 的 表单 
e 如 何 创建 分 别 具 有 显示 、 插 入 、 删 除 、 查 询 功能 的 脚本 

> 自我 检查 
下 面 检查 一 下 本 章 学 习 的 内 容 是 否 全 部 理解 并 掌握 了 。 





口 能 够 理解 使 用 PHP 和 MySQL 创建 Web 应 用 程序 的 过 程 
口 能 够 自制 一 个 简易 公告 板 


i 练习 题 





问题 1 

















按照 如 下 方式 将 本 章 介 绍 的 simple_select.php、simple_insert.php、simple_delete.php 和 simple_ 
search.php 合并 到 文件 simple2_process.php 中 ， 并 创建 simple2.html 作为 其 首页 。 
@ simple2.html ( 首页 ) 































































































使 用 具有 hidden 属性 的 <input> 标签 对 19.1.1 节 的 simple.html 进行 改进 ， 发 送 表示 显示 功能 的 
字符 sel、 表 示 插 入 功能 的 字符 ins、 表 示 删 除 功能 的 字符 del1， 以 及 表示 查询 功能 的 字符 ser， 以 便 在 
simple2_process.php 中 执行 这 4 个 处 理 。 
@ simple2_process.php ( 执行 处 理 的 PHP 脚本 文件 ) 

该 PHP 文件 使 用 从 simple2.html 发 来 的 字符 sel、ins、del 和 ser, 通 过 switch...case... (一 16.6.4 
节 ) 进行 分 支 处 理 ， 从 而 对 记录 执行 显示 、 插 入 、 删 除 和 查询 的 处 理 。 





































































































人 将 4 个 脚本 合并 为 1 个 。 当 然 ， 答 案 不 止 1 种 。 大 家 可 以 按照 自己 的 方式 进行 设计 。 





ta 参考 答案 








问题 1 








Q@ 创建 如 代码 清单 19-6 所 示 的 HTML 文件 simple2.html。 
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代码 清单 19-6 示例 simple2.html 





<!DOCTYPE html> 

<html> 

<head> 

<meta charset="UTF-8"> 
</head> 

<body> 


<p> 

<form method="POST" action="simple2 process.php"> 
<div> 显示 消息 </div> 

<input type="submit" value=" 显示 消息 "> 

<input type="hidden" name="h" value="sel"> 
</form> 


/BS 


去 四 天 > 


二 所 > 

<form method="POST" action="simple2 process. php"> 

<div> 输入 姓名 <input type="text" name="al"></div> 
<div> 输入 消息 <input type="text" name="a2" size=150></div> 
<input type="submit" value=" 发 送 消息 "> 

<input type="hidden" name="h" value="ins"> 

</form> 

</p> 


<hr> 


<p> 

<form method="POST" action="simple2 process.php"> 
<div> 输入 要 删除 的 编号 <input type="text" name="bl"></div> 
<input type="submit" value=" 发 送 删除 编号 "> 

<input type="hidden" name="h" value="del"> 

</form> 


</p> 


<hiES 


Pp 

<form method="POST" action="simple2 process.php"> 
<div> 输入 查询 的 关键 字 <input type="text" name="c1l"></div> 
<input type="submit" value=" 查询 "> 

<input type="hidden" name="h" value="ser"> 

</form> 


</p> 


</body> 
</html> 
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@ 创建 如 代码 清单 19-7 所 示 的 脚本 simple2_process.php。 


代码 清单 19-7 示例 simple2_process.php 





<?php 


/类 类 炎炎 炎炎 火炎 灾 炎 炎炎 炎 六 炎 连接 数据 库 、 选择 数据 库 淡淡 火 火炎 火炎 火炎 火炎 火炎 火炎 / 
$s=new PDO("mysql:host=localhost;dbname=dbl", "root", "root").; 
YSG 








/文大 太太 炎炎 太太 太太 大 大 大 xx 将 mame 为 hh 的 value 赋 给 Sh 加 ** 类 火炎 火炎 火炎 火炎 火炎 炎炎/ 
Sh dd BOSTLD"M]: 

















/汪汪 业 炎 炎炎 次 次 炎炎 次 交 次 次 次 $h_qd 根据 sel、 ins、del、ser 的 值 进行 条 件 分 支 处 理 。**** 炎 火炎 火炎 火炎 火炎 火炎 / 


switch("s$h ad"){ 
GASE SEL': 
Sre=$s->query ("SELECT * FROM tbk ORDER BY empid"); 
break; 
Case "ins™":; 
$al d=$ POST["al"]; 
$a2 d= $ POST["a2"]; 





$s->query ("INSERT INTO tbk (name,mess) VALUES ('$al d','$a2 d')"); 
Sre=$s->query ("SELECT * FROM tbk ORDER BY empid"); 
break; 


case, del 
$b1 d=$ POST["b1" 
$s->query ("DELETE FROM tbk WHERE empid=$b1 d"); 
$re=$s->query ("SELECT * FROM tbk ORDER BY empid"); 
break; 

CEASE "Sr" 
$c1 d=$ POST["c1" 
$re=$s->query ("SELECT * FROM tbk WHERE mess LIKE '%$c1 d%' ORDER BY 

empid"); 





break; 


/天 汪 炎 业 炎 炎 类 炎 业 炎炎 类 类 类 类 显示 查询 结果 六 炎炎 炎炎 炎炎 次 炎炎 炎炎 克 大大 了/ 


while ($result=$re->fetch()){ 
print $result [0]; 
询 冯 主攻 二 放 
print $result [1]; 
eh ly ot Le 时光 
print $result [2]; 
Beint. "cbr 


} 


类 火炎 火 火 火 火 火 火 火 火 火 火 火 火 口 的 类 火 火 火 火 火 火 火 火 火 火 火炎 火炎 
人/ 外向 首页 的 链接 / 






































print "<br><a href='simple2.html'> 返回 首页 </a>"; 














?> 
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注意 的 地 方 


有 些 读者 可 能 想 立 刻 使 用 前 面 学 到 的 知识 ， 将 Web 应 用 程 
序 部 署 到 租赁 服务 器 上 进行 发 布 。 但 是 在 此 之 前 ， 我 们 还 有 一 些 
知识 需要 掌握 。 
闫 网 上 穿行 着 很 多 卫生 人 ， 如 果 没 有 在 各 个 方面 做 好 防 
范 ， 数 据 库 就 会 面临 危险 。 

要 想 在 互联 网 上 发 布 Web 应 用 程序 ， 就 需要 采取 相应 的 防 
护 措施 。 本 章 我 们 将 学 习 安全 运用 Web 应 用 程序 所 需要 的 最 人 
限度 的 知识 。 




















































































































































































































20.1 不 在 发 布 的 文件 夹 中 放置 重要 信息 


六 20.1.1 PHP 文件 的 结构 

在 发 布 Web 页 面 的 时 候 ， 文 件 会 保存 到 发 布 在 Web 服务 器 上 的 文件 夹 中 (一 15.7.2 节 )。 这 
样 ， 任 何 可 以 连接 到 互联 网 的 人 都 可 以 下 载 已 发 布 的 文件 。 

那么 ，18.1.3 节 介 绍 的 那 种 操作 数据 库 的 脚本 也 会 被 连接 到 互联 网 的 人 下 载 吗 ?如果 任何 人 都 
可 以 自由 地 下 载 并 查看 其 内 容 ， 那 么 服务 器 名 、 用 户 名 ， 甚 至 密码 都 会 公之于众 。 

实际 上 ， 如 果 在 发 布 的 文件 夹 中 进行 了 PHP 运行 设置 ， 只 要 访问 扩展 名 为 “.php” 的 文件 ， 肢 
本 内 容 就 会 被 执行 ， 而 且 只 有 脚本 执行 的 结果 会 返回 给 访问 PHP 脚本 文件 的 客户 端 。 也 就 是 说 ， 
扩展 名 为 “php” 的 文件 的 内 容 不 会 被 公开 。 

此 外 ， 如 果 不 想 让 他 人 看 到 服务 器 上 的 文件 内 容 ， 可 以 通过 更 改 属性 来 限制 访问 。 

对 于 包含 重要 内 容 的 文件 ， 即 使 使 用 了 PHP 脚本 并 设置 了 访问 限制 也 不 可 掉以轻心 。 如 果 构 
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成 Web 应 用 程序 的 系统 中 存在 安全 漏洞 ， 那 么 本 来 不 可 见 的 文件 ， 有 时 也 可 能 出 于 某 种 原因 而 变 
得 可 见 。 更 何况 还 有 一 些 不 怀 好 意 的 人 。 所 以 ,把 写 入 了 密码 等 重要 信息 的 文件 放 在 发 布 的 文件 夹 
中 是 非常 危险 的 。 

那么 ， 当 使 用 脚本 处 理 重 要 信息 时 ， 我 们 该 怎么 办 才 好 呢 ? 























®20.1.2 如 何 读 取 其 他 文件 的 脚本 

我 们 可 以 将 作为 Web 服务 器 的 计算 机 设置 为 不 对 文件 夹 进行 发 布 。 包 含 重 要 信息 的 文件 必须 
放 在 难以 猜 到 其 位 置 的 私有 文件 夹 中 。 放 置 在 发 布 文件 夹 中 的 文件 不 要 包含 重要 的 信息 ， 需 要 用 到 
的 信息 从 其 他 地 方 读 入 即 可 。 


























区 require_once 
PHP 脚本 中 准备 了 一 些 读 取 其 他 PHP 脚本 文件 的 功能 。 本 节 我 们 将 使 用 require_once 命令 作 
为 示例 。 在 使 用 该 命令 读 取 文件 的 情况 下 ， 文 件 只 会 被 读 取 一 次 ， 如 果 无 法 读 取 文件 ， 则 停止 处 理 。 


require_once 








首先 ， 创 建 一 个 记述 了 服务 器 名 、 用 户 名 、 密 码 和 数据 库 名 的 文件 ， 并 将 此 文件 保存 在 不 被 发 
布 的 文件 夹 中 。 

创建 一 个 像 代码 清单 20-1 那样 的 文件 ， 并 将 文件 命名 为 db_info.php， 然 后 保存 在 不 被 发 布 的 
文件 夹 data 中 。 


代码 清单 20-1 db_info.php 








<?php 
$SERV="localhost"; 
$USER="root"; 
$PASS="root"; 
$SDBNM= "db1"; 


?> 























修改 19.3.2 节 连 接 数 据 库 的 脚本 simple_select.php， 使 其 能 够 读 取保 存 了 基本 信息 的 文件 db_ 
info.php。 
只 修改 连接 到 数据 库 的 部 分 即 可 ， 具 体内 容 如 下 所 示 。 


P 修改 前 





$s=new PDO("mysql:host=localhost;dbname=dbl","root","root"); 
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> 修改 后 


require once("data/db info.php"); 
$s=new PDO("mysql:host=$SERV;dbname=$DBNM", $USER, $PASS); 


require once ("data/db info.php") 用 于 读 取 data 文件 夹 中 的 db_info.php。 

为 了 便于 理解 ， 我 们 暂且 在 发 布 的 文件 夹 (htdocs 等 ) 中 创建 了 一 个 data 文件 夹 ， 并 在 其 中 放 
置 了 db_info.php。 当 然 ,， 保存 了 密码 等 敏感 信息 的 文件 ， 其 文件 夹 原本 应 该 放置 在 发 布 文件 夹 的 上 
一 级 ， 即 放置 在 不 被 公开 的 安全 场所 。 

修改 后 的 脚本 如 代码 清单 20-2 所 示 。 


代码 清单 20-2 simple_select2.php 




















<?php 
require once("data/db info.php"); 
$s=new PDO("mysql:host=$SERV; dbname=$DBNM", SUSER, $PASS); 


$re=$s->query ("SELECT * FROM tbk ORDER BY empid"); 
while ($result=$re->fetch()){ 

print $result [0] ; 

六 

Brint Sresult ll] 

Drihnt vs Vy 

print $result [2] ; 

DEL Mbe>" 


} 














print "<br><a href='simple.html'> 返回 首页 </a>"; 

















?> 











我 们 只 要 把 上 述 修改 操作 当成 将 读 取 的 文件 内 容 择 入 到 “require once ("data/db_ 
info.php") ;” 的 部 分 即 可 。 

如 果 对 simple_insert.php、simple_delete.php 和 simple_search.php 也 执行 相同 的 修改 ， 那 么 即使 
能 看 到 脚本 的 内 容 ， 也 不 会 看 到 密码 等 敏感 信息 了 。 











读 取 外 部 文件 的 命令 
除了 require once 之 外 ，PHP 脚本 中 还 有 几 个 读 取 外 部 文件 的 命令 。 例 如 include_once 命令 只 会 读 
取 一 次 文件 ， 即 使 读 取 文 件 失败 也 会 继续 处 理 。 














中 include_once 和 require once 的 区 别 体现 在 对 错误 的 处 理 方面 ，include_once 会 试图 导入 指定 的 文件 ， 即 使 
该 文件 没有 被 找到 ， 程 序 也 会 执行 ，require once 则 必须 导入 指定 文件 ， 如 果 没有 找到 该 文件 ， 程 序 则 不 会 继续 执 
行 。 另外， 引用 文件 的 命令 中 有 require、include 命令 ，_once 的 含义 在 于 判断 指定 文件 在 之 前 是 否 已 经 被 包含 
过 ， 如 已 包含 ， 则 忽略 本 次 包含 ， 确 保 它 只 被 包含 一 次 ， 以 免 出 现 函 数 重 定义 、 变 量 重 新 赋值 等 问题 。 译 者 注 
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20.2 ”避免 在 查询 中 输入 非法 数据 


® 20.2.1 什么 是 SQL 注入 


看 不 到 密码 等 重要 的 信息 就 代表 PHP+MySQL 是 绝对 安全 的 吗 ? 事实 上 并 非 如 此 。 更 可 怕 的 反 
而 是 SQL 注入 。 

“注入 ”( injection ) 一 词 一 般 用 于 表示 “注射 ”或 “投入 资金 "。 也 就 是 说 ，SQL 注入 表示 “ 注 
入 ”SQL 语句 。 这 是 一 种 通过 混入 (注入 ) Web 应 用 程序 开发 者 预想 之 外 的 数据 来 进行 非法 处 理 的 
攻击 方法 。 

接 下 来 我 们 就 来 体验 一 下 SQL 注入 的 恐 师 之 处 。 

请 大 家 回想 一 下 19.3.4 节 介 绍 的 PHP 脚本 simple_delete.php。 该 脚本 可 以 删除 指定 编号 的 记 
录 。 下 面 是 相应 的 部 分 。 















































Sbl_ d=$ POSTI["b1"]; 
$s->query ("DELETE FROM tbk WHERE empid=$b1 d"); 
$re=$s->query ("SELECT * FROM tbk ORDER BY empid"); 





在 发 送 方 simple.html 中 ， 把 要 删除 的 记录 编号 输入 到 <input . . .name="b1"> 文本 框 中 进行 
SUbmit。 

然后 ， 接 收 方 simple_delete.php 通过 $_PosT [ "bl" ] 接收 编号 ， 再 通过 $bl d=$_POST 
[bi" |] 将 编号 赋 给 $bl1_q， 并 以 此 为 基础 执行 如 下 查询 。 











DELETE FROM tbk WHERE empid=$bl1 d 


作为 开发 者 ， 我 们 可 以 预想 到 表 tbk 中 的 “6” 或 “12” 等 记录 编号 会 赋 给 $bl_q。 





Pp 体验 SQL 注入 


我 们 来 思考 一 下 ， 如 果 在 simple.html 的 用 于 输入 删除 编号 的 文本 框 中 输入 如 下 内 容 ( 见 图 20-1 )， 
并 点 击 “ 发 送 消息 ”按钮 ， 会 出 现 什么 样 的 情况 呢 ? 



































OR l= 


第 1 个 “1” 表 示 删 除 第 1 个 记录 。 但 是 ， 之 后 的 “oR 1=1” 又 是 什么 呢 ? 
将 oR 1=1 输入 到 前 面 的 SQL 语句 中 就 会 变 成 下 面 这 样 。 




















DELETE FROM tbk WHERE empid=1 OR 1=1 
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有 的 读者 可 能 已 经 注意 到 了 ,“1=1” 是 绝对 正确 的 (1 和 1 是 相同 的 ) 。 因此， 条件“ 当 
empid 等 于 1 的 时 候 , 或 者 当 1=1 的 时 候 ” 就 表示 “所 有 记录 都 满足 条 件 ”。 
也 就 是 说 ， 不 仅 是 第 1 个 记录 会 被 删除 ， 表 tbk 中 所 有 的 记录 都 会 被 删除 ( 见 图 20-2 )。 


一 [DD x | 
CW ES reocaossineenin St NA 
[Sern “ 故 












































输入 要 删除 的 编号 [TOR 1=7 @— | 输入 "1 OR 1=1” 
发 送 删除 编号 























输入 查询 的 关键 字 | 
坦 询 

















图 20-1 当 输入 “1 OR 1=1” 时 


x 
¢ /© ET -6 Sa 


古 localhost x 











近 四 首页 @ 所 有 记录 都 被 删除 了 








图 20-2 执行 结果 

实际 上 ， 在 本 示例 中 ， 如 果 发 送 “1 oR 1=1”， 所 有 的 记录 都 会 被 删除 。 这 样 就 很 容易 发 送 
Web 应 用 程序 开发 者 预想 之 外 的 危险 查询 。 

如 上 所 述 ， 通 过 发 送 包 含 了 部 分 SQL 语句 的 数据 来 攻击 系统 的 方法 就 是 SQL 注入 。 

不 过 ,第 19 章 介绍 的 脚本 simple_delete.php 原本 就 拥有 “可 随意 执行 删除 ”的 机 制 。 一 条 一 
条 删除 记录 和 一 次 性 删除 所 有 记录 是 一 样 的 ， 不 会 有 任何 损害 。 可 如 果 这 个 脚本 是 数据 库 的 重要 部 
分 ， 并 且 会 发 布 到 互联 网 上 实际 运行 ， 又 会 出 现 什么 样 的 情况 呢 ? 在 商务 交易 中 丢失 了 重要 的 顾客 
数据 就 糟糕 了 。 

在 现实 世界 中 ， 有 人 会 使 用 更 加 巧妙 的 SQL 注入 来 进行 各 种 各 样 的 攻击 。 即 使 完全 看 不 到 脚 
本 的 内 容 和 变量 ,他们 也 可 以 推测 出 变量 和 处 理 流程 ， 注 入 危险 的 SQL 语句 的 片段 。 

那么 在 发 布 Web 应 用 程序 的 时 候 ， 我 们 应 该 采取 什么 样 的 措施 呢 ? 
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> 禁止 输入 非 数字 值 
拿 上 面 的 示例 来 说 ， 只 要 创建 一 个 “如 果 发 送 了 数字 以 外 的 数据 ， 就 不 执行 DELETE” 的 机 钼 
就 可 以 了 。 发 送 的 数据 是 要 删除 的 记录 编号 ， 所 以 在 包含 字母 的 情况 下 不 执行 删除 ， 而 是 发 送 警 告 


























一 司 














检查 是 否 包含 字母 的 方法 有 很 多 种 ， 本 章 将 介绍 正则 表达 式 和 preg_match 函数 这 两 种 方法 。 


20.3 ”正则 表达 式 


全 20.3.1 什么 是 正则 表达 式 
正则 表达 式 是 一 种 用 于 描述 字符 排列 模式 的 方法 。 例 如 ， 表 达 式 [ 0-9 ] 表示 “包含 0 到 9 的 
数字 ”。 








本 节 将 介绍 下 面 这 些 典 型 的 示例 。 










































































> 包含 [ ] 中 的 字符 

[7] 包含 了 7 

[o-9] 包含 了 数字 

[a-z] 包含 了 小 写字 和 母 

[RA-z] 包含 了 大 写字 母 

[LA-2Za-z] 包含 了 大 写字 母 或 者 小 写字 下 

[LA-z] Lo-9] 头 是 大 写字 母 ， 之 后 是 数字 这 种 连续 字符 的 模式 





























P ”包含 除 [ ] 中 指定 的 字符 以 外 的 字符 


正则 表达 式 内 容 


























[0-9] 包含 除 0 到 9 以 外 的 字符 ( 不 包含 数字 ) 
[^a] 包含 除 A 以 外 的 字符 

[“a-2Z] 包含 除 大 写字 母 以 外 的 字符 

[ “0-9a-zA-2] 包含 除数 字 和 字母 以 外 的 字符 
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b> 以 “的 下 一 个 字符 开头 


^h 以 h 开 头 





”以 $ 的 前 一 个 字符 结尾 





ES 以 E 结尾 








7 {3 连续 出 现 了 3 次 以 上 的 7 





C 20.3.3 ”preg_match 函数 


preg_match 函数 是 一 个 使 用 正则 表达 式 进行 模糊 查询 的 函数 。 模 糊 查 询 不 是 瞄准 某 一 个 对 象 
进行 查询 ， 而 是 查找 具有 某 种 特征 的 字符 ， 从 而 得 出 更 多 检索 结果 的 一 种 查询 方式 。 
pregb_match 因数 如 下 所 示 。 








preg_match 函数 


; preg_match ( 正则 表达 式 ， 要 查找 的 字符 


HH 
UU 








preg_match 函数 使 用 与 脚本 语言 Perl 兼容 的 正则 表达 式 进行 查询 。 
在 这 种 情况 下 ， 正 则 表达 式 通常 需要 用 “/” 括 起 来 。 例 如 ,“ 包 含 大 写字 母 或 小 写字 母 ”的 正 
则 表达 式 是 [A-za-z ]， 那 么 表示 “包含 所 有 的 字母 ”的 正则 表达 式 就 是 下 面 这 样 。 














/[A-Za-z]/ 


代码 清单 20-3 使 用 了 正则 表达 式 / [A-2a-z] /来 检查 字符 串 “1234”。 因 为 “1234” 中 不 包 
含 字 母 ， 所 以 结果 会 显示 “不 包含 ”。 


代码 清单 20-3 include.php 








<?php 

if (preg match("/ [A-2a-2]/","1234")){ 
BRENtE 开 包含 "; 

}elsef{ 
print "不 包含 "; 


} 
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20.3.4 ”使 用 正则 表达 式 检 查 非 法 输入 
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Co 


再 介绍 一 个 使 用 了 正规 表达 式 的 示例 。 试 着 检查 输入 的 邮编 的 格式 是 否 为 “x x x-x x 
3“ 以 3 个 0 ~ 9 的 数字 开始 ”一 “-” 一 “以 4 个 0 ~ 9 的 数字 结束 ” 
用 于 判断 是 否 符合 上 述 字符 排列 顺序 的 表达 式 如 下 。 


“3 个 0 ~ 9 的 数字 ”一 [0-9]{13} 
“开始 ”一 “ 

“4 个 0 ~ 9 的 数字 ”一 [0-9] {4} 
“结束 ”一 ($) 


(VV VY 


代码 清单 20-4 是 对 变量 $m 的 内 容 107-0052 执行 检查 的 示例 。 


代码 清单 20-4 regular.php 


79 
XX o 





<?php 

$sm="107-0052"; 

if (preg match("/^[0-9] {3}-[0-9] {4}$/", $m)){ 
print " 暂且 是 OK 的 "; 

}else{ 
print " 有 错误 "; 























} 


EE 











107-0052 符合 条 件 ， 所 以 结果 会 显示 为 “ 暂 晶 是 OK 的 ”。 
当然 ， 要 想 在 互联 网 上 发 布 ， 仅 仅 这 么 做 是 不 够 的 。 为 了 进行 更 为 严密 的 查询 ， 我 们 还 
加 各 种 正则 表达 式 。 





”如果 不 是 数字 则 不 执行 查询 

首先 试 着 对 19.3.4 节 介 绍 的 “$s->query ("DELETE FROM tbk WHERE empid 
qd" ) ;” 进 行 重 写 ， 当 输入 数字 以 外 的 字符 时 ， 不 执行 DELETE 命令 并 显示 警告 。 
具体 来 说 就 是 当 $b1_a 的 内 容 中 包含 数字 以 外 的 字符 时 ， 通 过 输出 以 下 命令 来 显示 警 
且 不 执行 DELETE 命令 。 
































print "<div style='color:red'> 不 要 输入 除数 字 之 外 的 内 容 ! ! </div>"; 





代码 清单 20-5 改进 了 simple_delete.php 中 执行 删除 操作 的 部 分 。 


-Sbl 





告 ; 和 并 
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代码 清单 20-5 ”simple_delete.php 的 部 分 内 容 





if (preg match("/[^0-9]/",$b1 d) ){ 

print "<div style='color:red'> 不 要 输入 除数 字 以 外 的 内 容 ! ! </div>"; 
}elsef{ 

$s->query ("DELETE FROM tbk WHERE empid=$bl1 d"); 


} 




















如 果 在 变量 $b1l_a 的 值 中 输入 了 字母 等 字符 (输入 0 到 9 以 外 的 值 )， 则 会 显示 消息 “不 要 输入 除 
数字 以 外 …… ”， 并 且 不 执行 包含 DELETE 命令 的 查询 ,通过 “<div style='color:red'> ~ </ 
div>";” 设 置 的 字符 串 “ 不 要 输入 除数 字 以 外 ……” 将 变 为 红色 (一 17.62 节 ) 

如 果 没 有 输入 字母 等 字符 ( 没有 输入 0 到 9 以 外 的 值 )， 则 会 通过 “ss ->query ("DELETE 
FROM tbk WHERE empid=$b1l d" ) ;” 来 执行 删除 操作 。 

当然 作为 万 全 之 策 ， 我 们 不 仅 要 检查 是 否 输入 了 除数 字 以 外 的 值 ， 还 要 严格 检查 输入 的 数字 是 





























SQL 注入 的 方式 多 种 多 样 ， 我 们 无 法 一 一 介绍 。 不 过 ， 上 述 机 制 已 经 足以 防止 混入 多 余 的 SQL 
语句 了 。 


20.4 不 执行 非 预 期 标签 


标签 的 处 理 也 是 制作 Web 应 用 程序 时 很 重要 的 一 点 。 

我 们 使 用 17.8.2 节 的 send.html 和 receive.php 进行 一 下 实验 吧 。 这 两 个 文件 的 处 理 机 制 是 ， 如 
在 send.html 的 文本 框 中 输入 字符 并 点 击 “ 发 送 ” 按 钮 ，receive.php 就 会 使 用 print 输出 字符 。 

确保 receive.php 能 够 正常 运行 ， 然 后 显示 localhost 中 的 send.html。 在 文本 框 中 输入 如 下 所 示 
的 <body bgcolor=black>， 并 点 击 “ 发 送 ”按钮 ( 见 图 20-3 )。 

































































<body bgcolor=black> 














结果 会 是 什么 样 的 呢 ? 
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| a ln) 
Oo @ httpi//localhost/sendihtm EE D- ere} 











鲁 localhost Eh 


ET 
li 输入 <body bgcolor=black> 












































图 20-3 执行 send.html 
如 图 20-4 所 示 ， 窗 口 突然 变 黑 了 ! 到 底 发 生 了 什么 事 呢 ? 

















‘i S httpi//ocalhost/receive php 
优 localhost 








图 20-4 执行 结果 ( receive.php ) 
send.html 只 负责 将 输入 到 文本 框 中 的 字符 发 送 给 receivephp， 而 receive.php 只 负责 通过 print 输出 
接收 到 的 字符 串 。 也 就 是 说 ， 在 变 黑 的 浏览 器 画面 上 输出 的 是 <boqy bgcolor=black>。<body> 标 
签 的 bgcolor 属性 虽然 已 经 不 能 在 HTML 5 中 使 用 了 ， 但 是 它 在 大 多 数 的 浏览 器 中 还 可 以 正常 工作 。 
send.html 、receive.php 这 样 的 Web 应 用 程序 会 根据 输入 的 标签 ， 执 行 标签 设 定 的 功能 。 如 果 只 
会 出 现 上 面 这 样 的 问题 倒 也 没什么 ， 但 如 果 输 入 的 标签 能 执行 脚本 ， 在 某 些 情况 下 就 会 造成 严重 的 
后 果 。 因 此 ， 要 想 发 布 到 互联 网 上 ， 必 须 采 取 万 全 的 措施 。 













































































20.4.2 ”漏洞 攻击 


如 上 所 述 ， 将 在 Web 页 面 上 输入 的 内 容 直 接 输出 到 画面 上 的 程序 是 危险 的 。 不 怀 好 意 的 人 会 
通过 不 断 发 送 脚本 使 用 户 在 无 意 之 中 执行 恶意 脚本 ( 见 图 20-5 )。 针 对 此 类 系统 漏洞 的 攻击 会 一 致 
持续 下 去 。 


@ 
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为 了 防止 出 现 这 样 的 问题 ， 我 们 需要 制作 出 能 够 避免 脚本 混入 的 Web 应 用 程序 。 








Web 应 用 程序 




















普通 用 户 


预定 执行 的 处 理 

















漏洞 


不 怀 好 意 的 用 户 发 现 漏洞 ! 
所 写 % 




















o 


20-5 漏洞 攻击 


将 输入 的 字符 直接 输出 到 Web 页 面 是 非常 危险 的 。 我 们 先 来 想 办 法 让 输出 字符 串 中 包含 的 标 
在 这 种 情况 下 ， 可 以 使 用 htmlspecialchars 函数 将 标签 等 特殊 字符 转换 为 其 他 字符 串 。 











htmlspecialchars 函数 





htmlspecialchars 函数 可 以 将 标签 等 特殊 字符 串 按照 表 20-1 进行 转换 。 


表 20-1 通过 htmlspecialchars 函数 执行 的 转换 


转换 对 象 字符 转换 后 























< &lt; 
> &gt; 
& &amp; 
本 &quot,; 
&#039; 























※ 但 是 “'”( 单 引 号 ) 的 转换 只 能 在 第 2 参数 指定 为 ENT_QUOTES 的 情况 下 进行 
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恶意 输入 中 经 常 出 现 “<”“>”“&g”“"” 和 “'” 等 符号 。htmlspecialchars 困 数 能 够 将 
这 些 字 符 串 转换 为 无 法 执行 相应 功能 的 字符 串 。 
试 着 使 用 htmlspecialchars 修改 一 下 17.8.3 节 中 的 receive.php。 具 体 如 代码 清单 20-6 所 示 。 


代码 清单 20-6 receive_safe.php ( 摘要 ) 








<?php 
print htmlspecialchars($ POST["a"]); 


?> 














上 述 代 码 只 是 对 保存 了 接收 值 的 变量 $_POST | "a" ] 执行 htmlspecialchars。 

修改 完 脚本 后 ,保存 脚本 并 再 次 执行 send.html。 在 文本 框 中 再 次 输入 <body bgcolor=black> 
并 点 击 “ 发 送 ” 按 钮 ， 这 次 会 出 现 什 么 样 的 结果 呢 ? 

如 图 20-6 所 示 ， 这 次 按 原样 显示 了 输入 的 标签 ， 也 就 是 禁用 了 标签 的 功能 。 









DB | @ napyocalhostreceive safe.php -0 A 
[San ”“ 国 


<body bgcolor=black> 





图 20-6 ”执行 结果 
图 20-7 显示 了 执行 结果 的 源 代码 。 可 以 看 到 “<” 和 “>” 分 别 变 成 了 “glt;” 和 “&gt;”。 


cin 
MD | r/ocalhosreceive safe ptp 2 站 六 六 
傅 localhost x 

DOM 控制 调 坟 程 络 上 [响应 乏 查 器 内 存 -Edge 加 | ?日 x 


在 文件 中 查找 (Ctrl+ 晶 





20-7 ”显示 源 代码 
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如 何 创 建安 全 的 脚本 
我 们 需要 采取 一 切 可 能 的 措施 来 抵御 对 Web 应 用 程序 的 攻击 。 那 么 具体 来 说 ， 要 采取 什么 样 的 措施 呢 ? 
可 采取 的 措施 主要 有 以 下 几 点 。 






























































@ 放 置 在 发 布 文件 夹 中 的 文件 要 尽 可 能 缩减 到 最 少 (一 20.1 节 )。 一 定 不 要 放置 密码 等 重要 的 数据 。 对 文件 
和 文件 夹 设置 访问 限制 
@ 创 建 一 个 能 够 严格 检查 输入 的 数据 ， 并 禁用 指定 格式 以 外 的 值 的 机 制 (一 20.2 节 ) 
@ 创 建 一 个 不 会 根据 发 送 的 值 执行 预 想 之 外 的 操作 的 机 制 


















































采取 了 上 述 措施 是 否 就 可 以 放心 了 呢 ? 并 非 如 此 。 因 为 针对 Web 应 用 程序 的 攻击 方法 每 天 都 在 进化 。 我 们 
平时 使 用 的 HTTP 协议 ( 一 15.3.2 节 ) 基本 上 执行 一 次 “请 求 一 响应 ”就 结束 了 。 像 这 种 没有 个 人 身份 验证 的 、 
任何 人 都 可 以 使 用 的 系统 ， 老 实说 很 难 采取 措施 来 彻底 抵御 攻击 。 

我 们 需要 考虑 最 坏 的 情况 ， 并 且 时 常 思 考 相 应 的 措施 。 


20.5 ”总结 


本 章 介绍 了 以 下 内 容 。 





























































































































@ 在 互联 网 上 发 布 Web 应 用 程序 时 需要 注意 的 地 方 
@ 包 含 密码 等 重要 信息 的 文件 的 处 理 方法 

@ 针 对 SQL 注入 采取 相应 措施 的 示例 

@ 针 对 混入 标签 这 种 类 型 的 攻击 采取 相应 措施 的 示例 


绝对 不 能 忽视 安全 措施 。 因 为 自 es 自己 产生 影响 ， 还 会 给 很 多 人 带 来 麻烦 。 





当然 ， 光 使 用 本 章 介绍 的 技术 还 不 够 ， 我 们 还 安全 防范 意识 ， 这 一 点 非常 重要 。 
> 自我 检查 

















下 面 检查 一 下 本 章 学 习 的 内 容 是 否 全 部 理解 并 掌握 了 。 





口 能 够 理解 什么 是 SQL 注入 

口 能 够 使 用 require_once 函数 读 取 其 他 文件 的 脚本 

口 掌握 禁用 标签 的 方法 

口 掌握 使 用 preg_match 函数 和 正则 表达 式 进行 检查 的 方法 
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名 


加 练习 是 








问题 1 


JavaScript 的 命令 可 以 记述 在 script 标签 中 。 











国 在 HTML 中 编写 JavaScript 的 标签 








. 
































另外 ， 在 JavaScript 中 ，alert ("消息 ") 可 以 用 于 显示 消息 对 话 框 。 请 使 用 这 些 功 能 ， 思 考 发 送 什 
么 样 的 内 容 才 能 让 19.1.1 节 的 simple.html 按照 如 下 方式 工作 。 











在 执行 JavaScript 的 客户 端 环境 中 ， 当 普通 用 户 访 问 simple.html 时 ， 会 显示 一 个 消息 对 话 框 。 


ta 参考 答案 


问题 1 














如 图 20-8 所 示 ， 在 文本 框 中 输入 如 下 消息 ， 点 击 “ 发 送 消 息 ” 按 钮 。 执 行 结果 如 图 20-9 所 示 。 





<script>alert ("消息 ")</script> 


品 x 
Fs | 
【< 6 于 
[ee ”而 




















输入 消息 
<Script>alert(" 非 法 处 理 的 实验 ")</script> 
发 送 消息 

















输入 要 删除 的 编号 
发 送 删 除 编号 


输入 查询 的 关键 字 | 
坦 询 



































20-8 输入 消息 
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RE 
OY) @ http://localhost/simple_insert.plB ~ X 上 搜索 … 
Se ry A 


显示 消息 
显示 消息 


输入 姓名 
输入 消息 
<Sscript>alert(" 非 法 处 理 的 实验 ")</script> 
发 送 消息 




























x 






































输入 要 删除 的 编号 | 
发 送 删除 编号 


输入 查询 的 关键 字 | 
坦 询 









































图 20-9 当 普 通用 户 访问 能 够 执行 脚本 的 环境 时 
注意 ， 当 浏览 器 设置 为 “禁用 活动 脚本 ”时 ，JavaScript 在 某 些 环境 中 可 能 无 法 正常 工作 "。 











中 我 们 可 以 在 卫 的 internet 选项 一 安全 选项 卡 一 自 定 义 级 别 一 脚本 中 启用 活动 脚本 。 译 者 注 


第 21 章 ”创建 一 个 实用 公告 板 














在 第 19 章 中 ， 我 们 试 着 制作 














了 一 个 简易 公告 板 。 本 章 我 们 











来 制作 一 个 























定 实用 性 的 公告 板 。 这 部 分 内 容 是 前 面 所 学 技 














术 的 总 结 。 有 些 地 方 可 能 会 比较 难 























， 请 大 家 查 一 下 各 个 描述 的 含 























义 ， 整 理 一 下 所 学 知识 。 


21.1 创建 一 个 实用 公告 板 








本 章 是 本 书 的 最 后 一 章 ， 这 


和 PHP 的 教材 ， 但 在 制作 内 部 网 的 公告 板 方 面 ， 本 书 介绍 的 知识 也 

















们 会 用 到 前 面 介 绍 的 所 有 知识 。 
这 些 代 码 的 含义 和 作用 。 注 释 中 











章 会 介绍 一 个 具有 





定 实用 性 的 公告 板 。 








虽然 本 书 是 用 于 学 习 MySQL 


能 得 到 充分 的 应 用 。 本 章 ， 我 





因为 各 处 都 添加 了 注释 ， 所 以 请 大 家 一 边 仔细 阅读 代码 ， 一 边 思考 





包含 了 今后 大 家 在 





实用 公告 板 由 以 下 PHP 脚本 文件 构成 。 





出 作 Web 应 用 程序 时 能 用 到 的 各 种 各 样 的 提示 。 

















一 bulletin_top.php. oooeceooooi 首页 

上 =— bulletin phpimai 各 主题 的 回帖 显示 
六 bulletin__search.php.……: 关键 字 查询 

产 一 bulletin_reset.php pp 初始 化 数据 





和 db_info.php 数据 库 的 各 种 信息 











i 首页 用 的 














图 上 
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P bulletin_top.php 


bulletin_top.php 是 公告 板 首页 的 脚本 文件 。 该 脚本 文件 的 执行 结果 如 图 21-1 所 示 。 首 页 会 显示 
当前 存在 的 主题 名 称 以 及 它 的 创建 日 期 和 时 间 。 主 题名 称 上 设置 了 超 链接 ， 点 击 超 链 接 会 跳 转 到 各 
个 主题 的 回帖 列表 。 

我 们 可 以 通过 输入 主题 名 称 来 创建 新 的 主题 。 

另外 ， 页 面 上 还 有 指向 信息 查询 页 面 bulletin_search.php 的 超 链接 。 














x 


* 国 介 妆 多 @ 











Ge @ http//localhost/bulletin_top.php 只 - C | 义 SQL 和 啡 厅 的 页 面 











图 21-1 bulletin_top.php 的 执行 结果 


P bulletin.php 


bulletin.php 是 用 于 显示 每 个 主题 回帖 的 脚本 文件 。 该 脚本 文件 的 执行 结果 如 图 21-2 所 示 。 页 
面 中 包括 相应 主题 内 的 所 有 回帖 内 容 〈 回帖 创建 者 的 名 称 、 创 建 日 期 、 消 息 )。 

我 们 可 以 在 显示 的 主题 上 进行 跟 帖 。 

页 面 上 有 一 个 指向 首页 “主题 列表 ”( bulletin_top.php ) 的 超 链接 。 
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@O Ehttp://localhost/bulletinph| P © 











一 x 
|@ sum r1 关 于 实用 .x|D | N&O 





21-2 bulletin.php 


Pb bulletin search.php 


输入 关键 字 后 ，bulletin_search.php 会 从 所 有 的 主题 中 查询 包含 这 个 关键 字 的 消息 ， 并 显示 结果 
列表 ( 见 图 21-3 )。 


页 面 上 有 一 个 指向 首页 “主题 列表 ”(bulletin top.php ) 的 超 链 接 。 




















@O @ http//localhost/bullet P O 


(查询 结果 如 下 》 
3 :高 桥 :我 非常 喜欢 狗 ， 特 别 是 柯 基 大 。( 宠物 的 感觉 


4: 渡 边 : 我 也 非常 喜欢 狗 。 非 常 着 迷 和 小 狗 一 起 玩 ~ ( 宠物 的 感觉 ) 











一 x 
| 称 SQLynn 厅 的 查询 和 面 “ x| 可 从 六 作 @ 








请 输入 消息 中 含有 的 字符 ! 
查询 字符 串 




















将 " 狗 ” 作 为 关键 字 进 行 查询 











图 21-3 bulletin_search.php 
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Pb bulletin_reset.php 














bulletin_reset.php 是 重 置 数据 页 面 的 脚本 文件 。 调 用 该 脚本 文件 会 删除 正在 使 用 的 两 个 表 的 记 
录 ， 并 初始 化 自动 连续 编号 功能 ( AUTO_INCREMENT=1 )。 
任何 地 方 都 没有 它 的 链接 ， 所 以 该 脚本 需要 单独 执行 。 











P db_info.php 





db_info.php 是 用 于 读 取 输入 的 服务 器 名 、 用 户 名 、 密 码 和 数据 库 名 的 脚本 文件 (一 21.6 节 )。 
这 里 我 们 把 它 保存 在 不 被 发 布 的 data 文件 夹 中 。 
用 到 的 功能 都 是 本 书 介绍 过 的 内 容 。 大 家 可 以 慢 慢 理解 这 个 公告 板 的 处 理 流程 ( 见 图 21-4 )。 

















® 显示 、 创 建 主题 。 关 键 字 查 询 e 初始 化 表 


we | bulleti | 
人 、 En bulletin_reset.php 


9 保存 服务 器 名 、 
密码 和 数据 库 名 



























































9 显示 、 创 建 区 


ee | ee | < db_info.php 


图 21-4 实用 公告 板 的 结构 概念 图 





21.1.1 ”准备 实用 公告 板 中 使 用 的 图 片 


这 个 公告 板 上 使 用 了 图 片 文 件 jk.gif。 请 将 下 载 示 例 的 to_htdocs 文件 夹 中 的 pic 文件 夹 复制 到 
发 布 的 文件 夹 ( C:\MAMP\htdocs ) 中 。 
另外 ， 即 使 没有 图 片 也 不 会 影响 运行 ， 我 们 也 可 以 设置 成 自己 喜欢 的 图 片 。 
































21.2 ”创建 实用 公告 板 中 使 用 的 表 


实用 公告 板 使 用 了 tbj0 和 tbjl 这 两 个 表 。 在 制作 PHP 脚本 之 前 ， 请 参考 4.4.3 节 创 建 这 两 个 表 。 





























eS 21.2.1 tbj0 ( 主题 表 ) 


表 tbj0 用 于 存储 各 主题 的 数据 。 
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> 表 tbj0 的 列 






























































































































































group_c 用 于 输入 主题 的 组 号 。 列 为 INT 类 型 且 具 有 自动 连续 编号 功能 ( 一 6.8.1 节 ) 

topic_c 用 于 输入 主题 名 ， 数 据 类 型 为 VARCHAR (30) 

dae 于 输入 创建 主题 的 日 期 和 和 时间。 通过 MySOQL 的 NOW 函数 (一 8.2.6 节 ) 自动 输入 。 数 据 类 型 为 
= DATETIME 

i 于 存储 发 送信 息 的 客户 端的 IP 地 址 。 不 显示 在 浏览 器 上 ， 而 是 作为 出 现 特殊 情况 时 的 记录 保留 

澳 下 来 。 这 里 暂 目 将 数据 类 型 设置 为 20 个 字符 的 字符 串 类 型 VARCHAR ( 20 ) 




















> ” 表 tbj0 的 构成 


列 名 group_c topic_c 








属性 AUTO INCREMENT VARCHAR ( 30 ) | DATETIME VARCHAR ( 20 ) 
PRIMARY KEY 








表 tbjl 用 于 存储 所 有 主题 的 回帖 。 


PP ” 表 tbj1 的 列 



























































































































































empid 用 于 存储 所 有 主题 中 回帖 的 编号 。 列 为 INT 类 型 且 具 有 自动 连续 编号 功能 
name 用 于 输入 执行 输入 操作 的 人 的 姓名 ( 操作 者 姓名 )。 数 据 类 型 为 VARCHAR ( 30 ) 
mess 用 于 输入 消息 。 数 据 类 型 为 TEXT 

date_c 于 输入 插入 记录 时 的 日 期 和 时 间 。 通 过 MySQL 的 NOW 函数 自动 输入 
group_c 于 存储 主题 的 编号 。 作 为 和 表 tbj0 连接 时 的 键 使 用 。 数 据 类 型 为 INT 

ip_c 和 表 tbj0 一 样 ， 用 于 存储 发 送信 息 的 客户 端的 IP 地址 






































PP 表 tbjl 的 构成 


mess group_c ip_c 








属性 AUTO_INCREMENT VARCHAR (30 ) TEXT DATETIME INT VARCHAR (20) 
PRIMARY KEY 
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21.3 ”制作 首页 ( 创建 主题 以 及 显示 列表 ) 


下 面 我 们 来 制作 首页 bulletin_top.php 吧 。 

当 访问 bulletin top.php 时 ， 主 题 列 表 会 显示 出 来 。 在 下 面 的 文本 框 中 输入 新 的 主题 名 称 ， 点 击 
按钮 ，bulletin top.php 就 会 调用 自身 并 创建 一 个 主题 。 

另外 ， 通 过 底部 的 链接 可 以 跳 转 到 查询 页 面 bulletin_search.php。 

因为 /**...**/ 是 注释 语句 ， 所 以 不 输入 也 没有 关系 。 

































































@213.1 bulletin_top.php 的 代码 清单 


代码 清单 21-1 bulletin_top.php 





<?php 


/炎炎 火炎 火炎 火炎 火炎 火炎 火炎 ” 读 取 数据 库 信息 等 天 天天 类 类 类 类 类 类 类 类 大/ 
require once("data/db info.php"); 

















/ 汪 炎 风 次 次 次 次 次 次 次 次 次 次 次 次 连接 数据 库 ， 选 择 数据 库 次 次 次 次 次 次 次 次 次 次 次 次 次 次 次 / 
$s=new pdo("mysql:host=$SERV; dbname=$DBNM", SUSER, $PASS)，; 








二 日 生 
/炎炎 火炎 火炎 火炎 火炎 火炎 火炎 显示 标题 、 图 片 等 。 交大 大 大 大 大 克 炎炎 炎炎 赤 克 赤 克 人 








print <<<eot1 
<!IDOCTYPE html> 























<html> 

<head> 

<metafoeharset=uULeE 8 
<title>SQL 咖啡 厅 的 页 面 </title> 

</head> 

<body style="background-color:silver"> 

<img src="pic/jk.gif" alt=" 女孩 的 插图 "> 











<span style="color:purple;font-size:35pt"> 
SQL 咖啡 厅 的 公告 板 哦 


</span> 














<p> 请 点 击 要 查看 的 主题 编号 </p> 

<hr> 

<div style="font-size:20pt"> ( 主题 列表 ) </div> 
eot1; 











/炎炎 火炎 火炎 火炎 火炎 火 获取 客户 端 IP 地 址 类 类 类 类 类 类 类 类 类 类 类 类 类 类 / 
$ip=getenv ("REMOTE ADDR"); 
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/xxxxxxxxxxxxxxx 如果 主题 名 称 的 变量 $su_d 中 有 数据 ， 则 将 其 插入 表 tbj0 www / 
Psu SSee (Te so > hemlspeciale hars (Ree So 回 
i (Sa GD { 

$s->query ("INSERT INTO tbj0 (topic c,date c,ip c) VALUES ('$su 
dniieow() "ip Ms 


} 








S$re=$s->query ("SELECT * FROM tbj0"); 
while ($result=$re->fetch()){ 
print <<<eot2 











<a href="bulletin.php?gu=$result [0] ">$result [0] $result [1]</a> 四 
<br> 
$result [2] 创建 <pr><br> 

eot2; 

} 

/ 玉 汪 汪汪 类 类 类 类 灾 灾 类 关 类 实 光 用 于 创建 主题 的 表单 ， 以 及 查询 页 面 的 链接 六 炎炎 火灾 炎炎 火炎 灾 炎 火炎 炎 太 / 
































print <<<eot3 
<hr> 
<div style="font-size:20pt"> (创建 主题 ) </div> 
请 在 这 里 创建 新 主题 ! 
<br> 
<form method="GET" action="bulletin top.php"> 
新 创建 主题 的 标题 


=mpeut ey Be ue name a > 




















<div><input type="submit" value=" 创建 "></div> 
</form> 

<hr> 

<span style="font-size:20pt"> ( 查询 消息 ) </span> 


<a href="bulletin search.php"> 点 击 这 里 查询 </a> 加 








hE> 

</body> 

</htimls 
eot3; 


?> 











> ”指定 字符 编码 ( 国 ) 


这 是 用 于 设置 页 面 字符 编码 的 <meta> 标签 (一 17.5 节 )。 在 本 书 使 用 的 环境 中 ， 字 符 编码 指 
定 为 UTF-8。 

















<meta charset="utf-8"> 
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P 插入 图 片 ( 回 ) 

使 用 <img> 标签 插入 图 片 〈 一 17.5 节 ) 使 用 src 属性 指定 图 片 文件 的 位 置 。 这 里 显示 的 是 
pic 文件 夹 中 的 水 .gif 文件 。 要 想 显 示 图 片 ， 就 需要 事先 将 图 片 文件 复制 到 pic 文件 夹 中 。 

正确 复制 之 后 ， 在 浏览 器 的 地 址 栏 中 输入 http://localhost/pic/jk.gif， 图 片 就 会 像 图 21-5 那样 显 
示 出 来 。 当 然 ， 也 可 以 使 用 自己 原创 的 图 片 。 















































<img src="pic/jk.gif" alt=" 女 孩 的 插图 "> 











21-5 http://localhost/pic/jk.gif 


> 保存 主题 名 称 ( 加) 

在 通过 <input > 标签 指定 name="sun 的 文本 框 ( 回 ) 中 输入 新 的 主题 名 称 。 点 击 这 个 表单 
中 的 “创建 ”按钮 ， 数 据 将 通过 GET 方 法 发 送 到 bulletin top.php。 此 时 ， 回 就 会 使 用 $_ 
GET ["su"] 接收 新 主题 名 称 的 字符 串 ， 并 将 其 赋 给 变量 $ssu_d。 为 了 防止 非法 输入 ,我 们 可 以 使 
用 htmlspecialchars 图 数 (一 20.4.3 节 ) 禁用 标签 。 

isset 是 用 于 确认 变量 是 否 被 设置 的 函数 。 如 果 作 为 参数 的 变量 已 被 设置 ， 则 返回 TRUE。 只 
要 发 送 数 据 ，$GET ["su"] 的 变量 就 会 被 设置 。 因 此 ， 这 部 分 代码 表示 在 $GET ["su"] 被 设置 的 
情况 下 ，bulletin top.php 给 自己 发 送 了 数据 ， 于 是 将 htmlspecialchars ($_GET["su"] ) 的 值 
赋 给 变量 $su_dq。 如 果 没 有 被 设置 ， 则 表明 没有 发 送 数据 ， 在 这 种 情况 下 就 将 变量 $su_q 设置 为 
nul1， 其 实 就 是 进行 了 如 下 处 理 。 









































if isset($ GET["su"]){ 

$su d=htmlspecialchars($ GET["su"]); 
}else{ 

$su d=null; 


Y 
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如 果 使 用 16.6.2 节 中 介绍 的 三 元 运算 符 对 上 述 内 容 进 行 简化 ， 代 码 就 可 以 编写 成 下 面 这样 。 请 
注意 “?” 和 “:” 的 用 法 。 




















mm 











$su d=isset($ GET["su"])? htmlspecialchars($ GET ["su"] ) :nul11， 


P 显示 指向 主题 的 链接 ( 四) 
使 用 href 属性 在 指向 bulletin.php 的 链接 中 添加 ?gu=$result [ 0 ]， 这 样 就 可 以 使 用 GET 
方法 发 送 组 号 了 。 

$result [0] 是 SELECT * FROM tbj0 显示 的 记录 中 第 1 列 group c (主题 的 组 号 ) 的 值 。 
$result [1] 是 第 2 列 topic c (主题 名 称 ) 的 值 。 

<a href="pbulletin.php?gu=$result [0 ]">sSresult [0] $result [1]</a> 用 于 
显示 $result [0] (组 号 ) 和 8$result [1]( 主 题名 )， 并 且 在 该 字符 串 上 设置 指向 带 组 号 的 
bulletin.php 的 链接 。 

例如 ， 当 访问 组 号 为 3 的 主题 时 ， 地 址 栏 中 会 显示 http://localhost/bulletin.php?gu=3。 

主题 列表 的 写 出 部 分 如 下 所 示 。 





















































指向 pulletin .php 的 链接 给 gu 添加 组 号 后 发 送 


| | 


<a href="bulletin.php?gu=$result [0] ">$result [0] $result [1I] </a> 





ED 
sresult [2] 创建 <br><br> 






































date_c ( 主题 的 创建 日 期 和 时 间 ) group_c ( 组 号 ) topic_c ( 主题 名 ) 











P 创建 新 的 主题 ( 回 ) 

form 元 素 是 (递归 地 ) 调用 bulletin_ top.php 脚本 并 发 送 数据 的 表单 。 在 name=' su' 的 输入 
框 中 输入 新 创建 的 主题 名 ， 然 后 点 击 “创建 ”按钮 (type="submit" )，bulletin top.php 就 会 使 用 
GET 方法 向 自身 发 送 数据 。 图 的 $_GET ["su"] 会 接收 发 送 的 数据 。 



































<form method="GET" action="bulletin top.php"> 


> 指向 查询 页 面 的 链接 ( 回 ) 
“<a href="bulletin search.php"> 点 击 这 里 查询 </a>” 是 指向 具有 查询 功能 的 PHP 
脚本 bulletin search.php 的 链接 。 
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@213.2 bulletin top.php 的 结构 





我 伞 
工作 的 。 








人 


了 来 详细 看 一 遍 bulletin top.php 的 结构 ( 见 图 21-6 )。 这 次 试 着 看 






































数据 库 名 、 密 码 等 信息 
9 





> 
Y 


qbarmkeosonp 














----| 读 入 连 








接 








认 一 下 各 个 脚本 是 怎样 





<?php 


require once("data/db info.php"); 





$s=new PDO("mysql:host=$SERV;dbname=$DBNM", $USER, $PASS)，; 


print <<<eot1 
<!IDOCTYPE html> 


<html> 
<head> 


<meta charset="UTF-8"> 


<title> 
</head> 





SQL 咖啡 厅 的 页 面 </title> 

















<body style="background-color:silver"> 
<img src="pic/jk.gif" alt=" 女孩 的 插图 "> 
<span style="color:purple;font-size:35pt"> 
SQL 咖啡 厅 的 公告 板 


</span> 














<p> 请 点 


<hr> 


<div style="font-size:20pt"> ( 主 


eotl1; 


获取 客户 端 TP 地 址 \ 


6 要 查看 的 主题 编号 </P> 


























$ip=getenv ("REMOTE ADDR"); 
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如 果 设 置 了 S$ GET ("eu") . | 主题 名 Ee 












5 name="su" 


获取 主题 名 称 












1 
1 
1 
@ 
$su d=isset ($ GET 
[2 
1 


赋 给 变量 $su_a 获取 值 


插入 主题 的 数据 v 


["su"])? htmlspecialchars ($_GET["su"] ) :null; 









































保存 到 数据 库 ! 
1 
1 


if (ssu dGd<>""){ 


$s->query ("INSERT INTO tbj0 (topic c,date c,ip c) VALUES 
(es 
和 本 








主题 名 称 | | 日 期 和 时 间 | “~--| ip 地 址 
如 果 有 主题 名 称 的 信息 v 















































D4 





指向 各 主题 处 理 脚本 
的 链接 





























加 到 链接 上 EO topieno (EE 











设置 主题 链接 








了 
! 将 gu 的 值 添 
; 
1 
I 
1 


print <<<eot2 @ @ 












<a href="bulletin.php?gu=$result [0] ">$result [0] $result[1]</a> 
@、 





oh 


$result [2] 创建 <br><br> 
eot2; 9 
1 
1 
1 











表 tbjo 的 aate_c ( 日 期 和 时 间 ) 表 tbjo 的 group_c (组 号 ) 
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开始 编写 用 于 创建 主题 的 表单 











print <<<eot3 


<hr> 


<div style="font-size:20pt"> (创建 主 
请 在 这 里 创建 新 主题 ! 


























<br> 


























r -| 发 送 表 单 4 r= -| 调用 这 个 脚本 自身 
































<form method="GET" action="bulletin top.php"> 
新 创建 主题 的 标题 


<input, LVDS EEE .Name su Sle 0 ye 所 














1 
<div><input type="submit" value=" 创建 "></div> | 
</form> : 
1 
1 


于 输入 主题 名 的 文本 框 name = su 

































































1 
1 
1 
用 于 发 送 的 按钮 、 4 
指向 查询 页 面 的 链接 \ 


<h¥s 





<span style="font-size:20pt"> ( 查询 消息 ) </span> 








消 
<a href="bulletin search.php"> 点 击 这 里 查询 </a> 
<hr> 
</body> 
</html> 
eot3; 


?> 








图 21-6 bulletin_top.php 的 结构 


21.4 制作 各 个 主题 的 页 面 ( 输入 回帖 和 显示 列表 ) 


我 们 要 创建 显示 各 个 主题 的 消息 页 面 bulletin.php。 

访问 此 页 面 就 能 看 到 相应 主题 的 消息 。 在 下 面 的 文本 
用 bulletin.php 写 和 消息。 

另外 ， 通 过 底部 的 链接 ， 页 面 可 以 跳 转 到 bulletin top.php。 

















[a 
bt 





P 输 入 姓名 和 消息 并 点 击 按钮 ， 就 会 调 
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21.41 bulletin.php 的 代码 清单 


代码 清单 21-2 bulletin.php 





<?php 


类 炎炎 火炎 炎炎 火炎 炎炎 炎炎 火炎 S 古人 全 属 类 炎炎 炎炎 炎炎 炎炎 炎炎 炎炎 炎炎 
/ 读 取 类 言 息 委 / 


require once("data/db info.php"); 








/ 汪 炎炎 次 次 次 次 次 次 次 次 次 次 次 连接 数据 库 ， 选择 数据 库 六 淡 风光 风风火火 风 类 类 类 类 类 类 / 
$s=new PDO("mysql:host=$SERV; dbname=$DBNM", $USER, $PASS); 














/天 业 炎 业 炎 类 类 火 类 炎炎 类 炎炎 获取 主题 的 组 号 (gu) ， 将 其 赋 给 SU GG。 类 炎炎 炎炎 炎炎 炎炎 炎炎 炎炎 炎 类 / 
Se Ee SO 











/六 类 交火 炎炎 炎 交 炎炎 炎炎 xxx 如果 Sgu qd 中 包含 数字 以 外 的 字符 ， 则 停止 处 理 。 ** 关 炎炎 炎 交 炎炎 炎炎 殉 炎 炎 人 
if(preg match("/[“^0-9]/",s$gu aq) ){ 
print <<<eot1 



































输入 了 非法 的 值 <br> 
<a href="bulletin top.php"> 请 点 击 这 里 回 到 主题 列表 </a> 
eot1; 











/* 关 火炎 炎炎 炎炎 光 炎 xwxx ”如 果 Sgu_q 中 不 包含 数字 以 外 的 字符 ， 则 按 普 通 值 处 理 x*x**x**xxxx*** 炎 入 六/ 
}elseif (preg match("/[0-9]/",s$gu aq) ){ 











/太太 太太 大 大 大 大 大 大 大 xx ”获取 姓名 和 消息 并 删除 标签 ”x**** 类 类 业 类 类 类 类 类 类 类 类 了/ 
$na d=isset ($ GET["na"])?htmlspecialchars($ GET["na"]) :null; 
Smene sse (em me empee le neams(S ei me :rl 








/六 类 类 类 交火 类 类 类 类 类 类 类 类 燃 获取 IP 地 址 类 类 类 类 类 类 类 类 类 类 类 类 类 了/ 


$ip=getenv ("REMOTE ADDR"); 





/* 关 炎炎 炎炎 炎炎 炎炎 炎 xxxx 显示 与 主题 组 号 ( gu ) 相 匹 配 的 记录 ** 交 炎炎 交火 炎炎 炎炎 炎炎 炎炎 人 
SFEe=SSEdqUueruSERCTEEODIEEC RPRROMREEDJONRRRRESEOUPDEEESSUEGO 有 
sresult=s$re->fetch(); 

















/大 类 大 大 大 大 炎 克 大 大 大火 大 xx 创建 显示 主题 内 容 的 字符 串 $topic C Com * 炙 火炎 火炎 灸 火 六 火炎 交 火 六 六 / 


$topic c¢ com=" |[".$gu d." ".$result[0] ." | "; 








/六 太太 太太 太太 六 六 六 六 六 六 输出 主题 显示 的 标题 交火 炎炎 炎炎 类 炎炎 火光 类 类 大大 / 





上 上 
HH 


print <<<eot2 
<!IDOCTYPE html> 
<html> 
<head> 
<meta charset="UTF-8"> 
<title>SQL 咖啡 厅 $topic_c_com 主题 </title> 
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</head> 

<body style="background-color:silver'"> 
<div style="color:purple;font-size:35pt"> 
$topic _c_com 主题 | 





</div> 

<br> 

<div style="font-size:18pt">$topic c_com 的 消息 </div> 
eot2; 


/交大 炎炎 炎炎 炎炎 炎炎 类 xx 2 如果 输 入 了 姓名 ( $na_qd )， 则 将 记录 揪 入 七 pj1 交 关 炎炎 炎炎 炎炎 炎炎 灾 炎 风灾 炎 人 
i (sna 名和 ET 
Sre=$s->query ("INSERT INTO tbjl VALUES (0 Sna d','s$me d',now(),sgu Q, Sip7)7) 7 


/兴业 类 火炎 炎炎 灾 炎 类 火炎 类 业 类 显示 水 平 线 六 炎炎 炎炎 炎炎 类 类 大 类 大 类 大大/ 


DELnt: Vhs 














/ 文 炎 炎炎 炎炎 炎炎 炎炎 炎炎 炎炎 克 ” 按 时 间 | 顺序 显示 回帖 数据 天 天 类 类 类 类 类 类 类 类 类 / 
$re=$s->query ("SELECT * FROM tbjl1 WHERE group c=$gu d ORDER BY date c"); 








$i=1; 
while ($result=$re->fetch()){ 


print "$i($result[0]):s$result([1] :$result [3] <br>"; 
pra l(t sot 加 
Brint Vv<br><bES"; 

$i++ 


print <<<eot3 
<hr> 
<div style="font-size:18pt"> 
请 在 这 里 向 $topic_c_com 中 写 消息 
</div> 














<form method="GET" action="bulletin.php"> 
<div> 姓 名 <input type="text" name="na"></div> 
消息 

<div> 


回 


<textarea name="me" rows="10" cols="70"></textarea> 
</div> 

<Imput type = uiclenu name Mou velue So ud 日 
<input type="submit" value=" 发 送 "> 

</form> 

<h¥ 

<a href="bulletin top.php"> 返回 主题 列表 </a> 

</body> 

</html> 
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ecot3 








/** 久 火炎 炎炎 类 交 六 六 交 当 S$gu qd 不 包含 数字 也 不 包含 数字 以 外 的 字符 时 的 处 理 交 交 交 交 赤 赤 炎炎 赤 炎 光 赤 赤 赤 赤 人 
}else{ 
print " 请 选择 主题 。<br>"; 











口 

















题 列表 </a>"; 





print "<a href='bulletin top.php'> 点 击 这 里 返 














区 获取 组 号 、 姓 名 和 消息 ( 辆 ) 

当 从 首页 bulletin_top.php 跳 转 到 bulletin.php 时 ，gu 会 通过 GET 方法 发 送 ; 当 从 bulletin.php 
跳 转 到 bulletin.php 时 ，gu、na 和 me 这 三 个 用 于 识别 数据 的 元 素 名 称 会 通过 GET 方法 发 送 。 贺 中 
代码 的 作用 就 是 通过 “$gu_d=$_GET ["gu"] ;” 接 收 这 些 数据 ， 并 将 其 赋 给 $Sgu_d 等 变量 。 另 
外 ， 表 tbjl 保存 了 所 有 回帖 ， 通 过 GET 方法 发 送 的 gu、na、me 是 表 tbjl 的 列 group_c、 列 name 
和 列 mess 中 的 数据 。 

对 于 姓名 (na ) 和 消息 (me )， 我 们 可 以 使 用 htmlspecialchars 函数 来 禁用 标签 ， 从 而 防 
止 非法 输入 。 另 外 ， 如 果 没 有 从 页 面 自 身 (bulletin.php ) 进行 跳 转 ， 就 表明 $_GET["na"] 和 8$ 
GET ["me"] 中 的 变量 没有 被 设置 。 如 果 变 量 被 设置 了 ， 就 表明 给 自己 (bulletin.php ) 发 送 了 数据 ， 
于 是 通过 isset 函数 (一 21.3.1 节 ) 将 htmlspecialchars ($8_GET["na"]) 的 值 赋 给 变量 
$na d, 将 htmlspecialchars($_GET["me"]) 的 值 赋 给 变量 sme dq。 如 果 变 量 没有 被 设置 ， 
则 表明 没有 发 送 数据 ， 在 这 种 情况 下 变量 将 被 设置 为 nul1。 





















































$gu d=$ GET["gu"]; 
$na d=isset($ GET["na"])?htmlspecialchars($ GET["na"]) :null; 
$me d=isset($ GET["me"])?htmlspecialchars($ GET["me"]) :null; 


Pp 发 送 SELECT 语句 ( 回 ) 
sgu_qd 是 主题 的 组 号 。 发 送 一 条 SQL 语句 ， 从 表 tbj0 中 获取 该 组 号 相应 的 主题 名 ( 列 topic_c )。 





$re=$s->query ("SELECT topic cC FROM tbj0 WHERE group c=$gu d"); 


P 获取 组 号 和 主题 名 ( 图 ) 


.$gu d 是 组 号 ，$result [0 |]. 是 列 topic_c 的 主题 名 。 将 这 两 个 字符 串 用 “. ”连接 起 来 的 


字符 串 是 Stopic_c_com。 
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> ”指定 字符 编码 ( 四) 
<meta> 标签 用 于 设置 字符 编码 。 在 本 书 使 用 的 环境 中 ， 字 符 编码 为 UTF-8。 


P 发 送 INSERT 语句 ( 回 ) 


向 保存 回帖 的 表 tbjl 的 列 empid、name 、mess、date c、group c 和 ip_c 中 分 别 输入 0、 姓 名 
(Sna_ da)、 消 息 (Sme_ Q)、 当 前 日 期 和 时 间 (now() )、 组 号 ($gu_d) 和 了 他 地 址 ($ip)。 如 果 
在 设置 了 AUTO_INCREMENT 的 列 empid 中 输入 0， 编号 就 会 自动 连续 输入 进去 (一 6.9 节 )。 


> 显示 消息 ( 回 ) 


$ result [2] 是 园 中 输入 的 消息 。 如 果 这 个 字符 串 中 有 换行 符 ， 可 通过 nl12br 函数 
(一 17.7.2 节 ) 使 实际 显示 的 内 容 实现 换行 。 


> 设置 文本 区 域 ( 园 ) 


这 里 第 一 次 出 现 了 <textarea> 标签 ， 这 个 标签 用 于 设置 可 以 实现 多 行 输入 的 文本 框 。 可 通 
过 name 设置 “用 于 识别 数据 的 元 素 名 称 ”， 通 过 rows 设置 行 数 ， 通 过 cols 设置 每 行 的 字符 数 。 


P 发 送 组 号 ( 回 ) 


如 果 在 type 属性 中 指定 了 hidden， 在 这 种 情况 下 虽然 不 会 在 浏览 器 中 显示 任何 内 容 ， 但 会 
发 送 value 属性 指定 的 值 。 本 例会 使 用 元 素 名 称 gu 自动 发 送 $gu_a 的 值 (组 号 )。 
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@2142 bulletinphp 的 绪 构 
我 们 再 来 详细 看 一 下 bulletin.php 的 结构 ( 见 图 21-7 )。 


数据 库 名 、 密 码 等 信息 | 





db_info.php 











<?php 


require once("data/db info.php"); 


$s=new PDO("mysql:host=$SERV;dbname=$DBNM", $USER, $PASS) ; ©@----' 
1 








连接 数据 库 


获取 name="gu" 的 值 v 
1 | 组 号 4-| name="gu" 















































由 



















































指向 首页 的 链接 
1 
1 
print <<<eot1 | 
输入 了 非法 的 值 <br> @ 
<a href="bulletin top.php"> 请 点 击 这 里 回 到 主题 列表 </a> 
SOE 


}elseif (preg match("/[0-9]/",$gu aq) ){ 
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给 变量 na_d、 


me_d 赋值 








me_d 已 被 设置 





如 果 变 量 na_d、 








$topic c com="| 























获取 




















| 

















$na d=isset ($ GET["na"])?htmlspecialchars($ GET["na"]) :null; 
$me d=isset($ GET["me"])?htmlspecialchars($ GET["me"]) :null; 
9 9 


一 rz 一 一 一 
从 





一 


Ee ed E4 
name="me" +? 7 和 | 

















相同 ” 


条 件 是 “与 发 送 的 组 号 ( gu ) 


























查询 的 主 











题名 称 






































print <<<eot2 


<!IDOCTYPE html> 
<html> 
<head> 


<meta charset="UTF-8"> 


<title>SQL 咖啡 厅 $topic _c_com 


</head> 





主题 </title> 


<body style="background-color:silver"> 


<div style="color:purple;font-size:35pt">$topic ¢ com 


<br> 


主题 | 


</div> 





<div style="font-size:18pt">$topic_c_com 的 消息 </div> 
ecot2 ; 
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如 果 $na_a 不 为 空 ( 如 果 输 入 了 姓名 } 、4 





























1 
1 
I INSERT 回帖 信息 





if ($na d<>""){ -| 

















@ 
$re=$s->query ("INSERT INTO tbj1 VALUES (0,'$na d','s$me_ 
d',now(),s$gu qd,'s$ip')"); ? 
} ® @ e@、 I | 
1 1 SS 1 1 
1 ~ 1 1 
1 RN 1 
当前 日 期 和 时 间 |*- 了 | 组 号 ~- IP 地 址 当 设 置 为 0 时， 可 以 | *--1 消息 






























































实现 自动 连续 编号 

















b 4 条 件 是 组 号 相同 








按照 日 期 和 时 间 排 序 








SELECT 回 帖 列表 
















@ @ 
$re=$s->query ("SELECT * FROM tbj1 WHERE group c=$gu d ORDER BY date c"); 


$i=1; 
while($result=$re->fetch()){ 








由 
EE 








8 换行 标签 








A 


@ 
print " 站 (5 [0] ) :$result 二 Se [31 <bEs Ty, 














' “~~-- 日 期 和 时 间 date_c 


























1 

a 1 
在 主题 内 显示 时 的 “1! | 所 有 回帖 的 编号 `、 
编号 empid \ 4] 姓名 name 
































BA 


print nl2br ($result [2] ); 
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回帖 的 编号 加 1 while 到 此 为 止 





由 
压 




















print <<<eot3 


<hr> 

xdiv style="fornt=size:18pt"> 
请 在 这 里 向 Stopic_c_com 中 写 消息 
</div> 














向 这 个 脚本 发 送 数 据 














<form method="GET" action="bulletin.php"> 


<div> 姓 名 <input type="text" name="na"></div> 
9 


于 输入 姓名 的 文本 框 vy 








轩 














-一 一 一 一 - 用 于 输入 消息 的 文本 框 














1 
消息 - 
LY | 
<textarea namea me” rows="10"®cols="70"></textarea> de 
上 ~ 
</div> spe 
~~~ 
1: 
































表单 的 最 后 name="me" 有 10 行 ， 每 行 70 个 字符 有 10 行 ， 每 行 70 个 字符 


动 发 送 $Sgu qd 的 值 y 


















































由 


<input type="hidden" name="gu" Value=$gu_ d> 
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设置 发 送 按钮 


<input type="submit" value=" 发 送 "> 


</form> 








1 
表单 结束 


设置 “返回 ”链接 


He 














D4 




















<a href="bulletin top.php"> 返回 主题 列表 </a> 
</body> 
</html> 
eot3; 
}else{ 
9. 








正式 处 理 结束 

















当 $gu_ qd 为 空 时 的 








print " 请 选择 主题 。<br>"; 


























print "<a href='bulletin top.php'> 点 击 这 里 Y 主题 列表 </a>"; 





21-7 ”bulletin.php 的 结构 


21.5 ”制作 消息 的 查询 页 面 


下 面 要 制作 的 是 从 所 有 主题 的 所 有 消息 中 执行 关键 字 查 询 的 页 面 bulletin_search.php。 在 文本 框 
中 输入 关键 字 并 点 击 按 钮 ，bulletin_search.php 就 会 调用 自身 执行 查询 ， 并 显示 提取 的 记录 。 
另外 ， 通 过 底部 链接 ， 页 面 可 以 返回 到 bulletin top.php。 
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@215.1 bulletin_search.php 的 代码 清单 


代码 清单 21-3 ”bulletin_search.php 





<?php 


/xxxxxx 炎 xxxxxxxx 读 取 数 据 库 信息 等 关 * 交 大 六 六 大 大 大 六 六 大 大 大 大/ 
require once("data/db info.php"); 





A/ 太 炎 炎 灾 灾 灾 灾 灾 灾 灾 灾 灾 灾 灾 灾 连接 数据 库 ， 选 择 数据 库 六 交 次 次 次 次 次 次 次 次 次 次 次 次 次 / 
$s=new PDO("mysql:host=$SERV; dbname=$DBNM", SUSER, $PASS); 











/天 业 火炎 火炎 火炎 火炎 火炎 炎炎 痰 显示 标题 等 六 火炎 炎炎 炎炎 炎炎 火炎 火 类 类 类 / 


print <<<eot1 
<!IDOCTYPE html> 




















<html> 
<head> 
<meta charset="UTF-8"> 
<title>SQL 咖啡 厅 的 查询 页 面 </title> 
</head> 
<body style="background-color:aqua"> 
hes 
<div style="font-size:18pt"> (查询 结果 如 下 ) </div> 
eot1,; 








/汪汪 炎炎 风灾 火炎 次 灾 次 灾 火炎 风 获取 查询 字符 串 并 删除 标签 六 炎炎 次 次 次 次 次 交 炎炎 次 交 六 风 / 
sserd=isset (Ss CELT[l"se"l) ntmlspecralechars(s CoEL[I"sen)) :no 加 














/汪汪 火炎 火炎 火炎 火 六 火炎 六 2 如果 查 询 字符 串 ($se qd) 中 有 数据 ， 则 执行 查询 处 理 xx 关 关 大 大 六 六 大 大 太太 大 大 大 人 
if ($se d<>"") { 








/类 炎炎 火炎 火炎 火炎 炎 火炎 六 查询 的 SQL 语句， 连接 表 be 和 表 tbj0 类 类 炎炎 火炎 火炎 炎炎 炎炎 类/ 
$str=<<<eot2 
SELECT tbjl.empid,tbjl.name,tbjl.mess,tbjo.topic c 
FROM tbj1l 
JOIN tbj0 
ON 








Eb up ec tol ou 
WHERE tbjl.mess LIKE "%$se d%" 


eot2; 


/天 类 炎炎 火炎 炎炎 火炎 类 炎炎 火炎 执行 查询 交火 炎炎 火炎 炎炎 火炎 火炎 炎炎 类 / 

$re=$s->query ($str); 

while ($result=$re->fetch()){ 
print " $result[0] : $result [1] : $result[2] ( $result[3] )"; 
PEIN MBES<bEST; 
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/** 炎 大 大 X 赤 炎 大 xxxxxx 用 于 输入 查询 字符 串 的 页 面 ， 以 及 指向 首页 的 链接 。 xxr 大 大 太太 关 大 大 六 太太 大大/ 


print <<<eot3 

















<hr> 

<div> 请 输入 消息 中 含有 的 字符 ! </div> 

<form method="GET" action="bulletin search.php"> 
查询 字符 串 

<input type="text" name="Sse"> 

<div> 

<input type="submit" value=" 查询 "> 

</div> 

</form> 

<br> 

<a href="bulletin top.php"> 返回 主题 列表 </a> 
</body> 

</html> 











P> ”指定 字符 编码 ( 圆 ) 
<meta> 标签 用 于 设置 字符 编码 。 在 本 书 使 用 的 环境 中 ， 字 符 编码 为 UTF-8。 


> 给 要 查询 的 关键 字 赋值 ( 四 ) 
这 部 分 代码 用 于 接收 通过 se 发 送 过 来 的 关键 字 ， 并 将 其 赋 给 变量 $se_q。 


P 发 送 SELECT 语句 (图) 

通过 连接 表 tbjl 与 tbj0 (一 10.2 节 )， 查 询 回 帖 的 编号 、 姓 名 、 消 息 和 主题 名 称 。 连 接 键 是 两 
个 表 的 组 号 group c。 变 量 $se_d 是 要 查询 的 关键 字 ， 如 果 在 它 的 前 后 加 上 “%”， 就 能 查询 包含 这 
个 关键 字 的 所 有 消息 。 
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@@21.5.2 bulletin_search.php 的 结构 


我 们 再 来 看 一 下 bulletin search.php 的 结构 ( 见 图 21-8 )。 











| 数据 库 名 、 密 码 等 信息 | 
7< 9 


从 1 
从 1 


,| 读 取 
























dblnfor te 








require once("data/db info.php"); 








$s=new PDO("mysql:host=$SERV; dbname=$DBNM", $USER, $PASS); 


写 出 用 于 查询 的 页 面 \\ 


print <<<eotl1 
<!DOCTYPE html> 
<html> 
<head> 





<meta charset="UTF-8"> 

<title> SQL 咖啡 厅 的 查询 页 面 </title> 
</head> 

<body style="background-color:aqua"> 




















<hr> 
<div style="font-size:18pt"> (查询 结果 如 下 ) </div> 
SOEtls 























设置 的 情况 下 


给 查询 关键 字 峰 值 | 1 


\ 
© ' 
$se d=isset ($ GET["se"])?htmlspecialchars($ GET["se"]) :null; 
9 9 


在 $_GET ("se") 被 | 查询 字符 串 + - name="sen 
一 一 一 一 一 



























给 变量 se_dq 赋 值 获取 值 


如 果 有 查询 关键 字 


if($se d<>""){ 




















21.5 制作 消息 的 查询 页 面 





连接 表 tbj1 和 表 tbj 0 
1 



















$str=<<<eot2 
SELECT tbjl.empid,tbjl.name,tbjl.mess,tbj0.topic c 
FROM tbjl 
JOIN tbj0 ® 
ON 
tbjl.group c=tbj0.group c®@ 
WHERE tbjl.mess LIKE "%$se ds%" 
eot2; 


























条 件 是 消息 中 包含 查询 字符 串 条 件 是 组 号 相同 


SELECT 语句 w 
































while($result=$re->fetch()){ 





















































编号 姓名 消息 主题 名 
T T T T 
1 1 1 1 
和 电 电 电 
print " $result[0] : $result[1] : $result[2] ( S$result [3] )"; 


print. "<brs<brs"; 





print <<<eot3 


<hes 
<div> 请 输入 消息 中 含有 的 字符 ! </div> 
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这 个 脚本 








v 调 
用 于 发 送 的 表单 N\ 





<form method="GET" action="bulletin sea 





b 


查询 字符 串 





<input type="text" name="Sse"> @------- 
<div> 

<input type="submit" value=" 查询 "> 
</div> 9 

</form> 








发 送 按钮 


Here Document 结束 


去 中 到 六 
<a href="bulletin top.php"> 返回 主题 列表 < 
</body> 
</html> 
eot3; 




















?> 





@ 
reh .php"»> 











































输入 查询 字符 串 的 文本 框 














/a 


图 21-8 ”bulletin_search.php 的 结构 


21.6 ”制作 读 取 数据 库 信息 的 原 





db_info.php 是 输入 了 服务 器 名 、 用 户 名 、 密 码 和 数据 库 名 的 文件 。 请 根据 自 





数据 。 这 里 ， 我 们 将 该 文件 保存 在 不 进行 发 布 的 data 文件 夹 


代码 清单 21-4 db_info.php 


始 文件 





ml 


[e) 









己 的 情况 适当 修改 





<?php 
$SSERV="localhost",; 
$USER="root"; 
S$PASS="root"; 
SDBNM="db1"; 


> 
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21.7 ”制作 数据 重 置 页 面 


我 们 来 制作 一 个 用 于 重 置 数据 的 页 面 。21.7.1 节 的 脚本 用 于 删除 表 tbj0 和 表 tbjl 中 的 所 有 记录 
(一 7.9 节 )， 并 重 置 AUTO_INCREMENT (一 6.8 节 )。 























@2171 bulletin_resetphp 的 代码 清单 


代码 清单 21-5 ”bulletin_reset.php 





<?php 
require once("data/db info.php"); 
$s=new PDO("mysql:host=$SERV; dbname=$DBNM", $USER, $PASS)，; 


$s->query ("DELETE FROM tbj0"); 
$s->query ("DELETE FROM tbj1"); 
$s->query ("ALTER TABLE tbj0 AUTO INCREMENT=1"); 
$s->query ("ALTER TABLE tbjl1 AUTO INCREMENT=1"); 











( 
( 
( 
( 


























print " 将 SQL 咖啡 厅 的 表 初 始 化 了 "; 


EE 











@2172 buletinresetphp 的 结构 


我 们 来 看 一 下 bulletin_reset.php 的 结构 ( 见 图 21-9 )。 








<?php 
require once("data/db info.php"); 
$s=new PDO("mysql:host=$SERV; dbname=$DBNM", $USER, SPASS) ; 








删除 所 有 记录 \ 


$s->query ("DELETE FROM tbj0"); 
$s->query ("DELETE FROM tbj1"); 
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初始 化 连续 编号 的 值 















$s->query ("ALTER TABLE tbj0 AUTO INCREMENT 
$s->query ("ALTER TABLE tbjl1 AUTO INCREMENT 









print" 将 SQL 咖啡 厅 的 表 初 始 化 了 "; 


Ee 





21-9 ”bulletin_reset.php 的 结构 


21.8 总结 


本 章 介 绍 了 以 下 内 容 。 


@ 实 用 公告 板 的 结构 

@ 主 题 处 理 的 功能 

@ 回 帖 处 理 的 功能 

@ 针 对 实用 公告 板 采取 的 最 低 限 度 的 安全 措施 


> 自我 检查 
下 面 检查 一 下 本 章 学 习 的 内 容 是 否 全 部 理解 并 掌握 了 。 




















口 了 解 以 bulletin_top.php 为 中 心 的 公告 板 系统 的 结构 
口 了 解 bulletin_top.php 的 处 理 程序 

口 了 解 bulletin.php 的 处 理 程序 

口 了 解 bulletin_search.php 的 处 理 程序 

口 了 解 表 tbj0 和 tbj1 的 初始 化 过 程 
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ta 练习 题 
问题 1 
请 制作 一 个 PHP 脚本 ， 该 脚本 可 以 将 正文 中 创建 的 实用 公告 板 中 使 用 的 表 tbj1 的 全 部 数据 按照 如 下 方 


式 输出 。 
























































实用 公告 板 制作 完成 了 。 
请 尽情 地 进行 输入 吧 。 


藕 育 啊 。 
会 尽情 地 进行 输入 的 哦 。 






次 狗 ， 特 别 是 柯 基 犬 。 






家 知道 青蛙 的 可 爱 吧 。 
我 现在 上 瘾 了 。 


我 从 知 床 回来 了 。 
黑 眼 圈 都 出 来 了 ， 大 家 不 要 效 夜 了 





@ http://localhost/bullei P ~ @ 感 Iocalhost 
1(1): 西 泽 :2019-08-24 06:14:03 GP:1 IP:127.0.0.1 


2(2): 中 川 :2019-08-24 06:14:52 GP:1 IP:127.0.0.1 
:2019-08-24 06:25:18 GP:2 IP:127.0.0.1 


019-08-24 06:25:54 GP:2 IP:127.0.0.1 
喜欢 狗 。 非 常 着 迷 和 小 狗 一 起 玩 ~ 


5(5): 松 田 :2019-08-24 07:52:07 GP:2 IP:127.0.0.1 


6(6): 西 泽 :2019-08-24 07:53:05 GP:4 IP:127.0.0.1 


x*[ 人 D3 








ta 参考 答案 























创建 代码 清单 21-6 中 的 PHP 脚本 。 


代码 清单 21-6 示例 bulletin_manage.php 








<?php 


/天 类 炎炎 炎炎 火炎 火炎 炎炎 炎炎 次 


读 取 数据 库 信 息 等 


require once("data/db info.php"); 





/天 汪 炎 炎炎 火炎 火炎 火炎 炎炎 炎炎 


连接 数据 库 ， 选 择 数 据 库 





六 炎炎 火炎 炎炎 炎炎 火炎 火炎 炎 类 / 


六 六 交 火炎 火炎 炎炎 炎炎 炎炎 类 类 / 


$s=new PDO("mysql:host=$SERV;dbname=$DBNM", $USER, SPASS) ; 














/兴业 炎炎 火炎 火炎 火炎 类 炎炎 炎炎 





对 表 tbj1 进行 SELECT 


交火 炎炎 炎炎 交火 炎炎 火炎 次 类 风 / 


$re=$s->query ("SELECT * FROM tbjl1 ORDER BY date c"); 


/天 类 炎炎 火炎 火炎 火炎 火炎 次 炎炎 
$i=1; 
while($result=$re->fetch()){ 





输出 查询 结果 


炎炎 火炎 火炎 炎炎 火炎 火炎 类 火炎 / 


print "$i(S$result[0]):$result [1] :$result [3] GP:S$result [4] 


IP:$Sresult [5] <br>"; 
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print nl2br ($result [2] ) ; 
DEiNt. Vbr><bres "> 
$1i++? 








使 用 phpMyAdmin 
常见 问题 的 检查 清 


MySQL 基 础 练习 





附录 1 使 用 phpMyAdmin 











phpMyAdmin 是 可 以 在 浏览 器 上 管理 MySQL 服务 器 的 应 
用 软件 。 如 果 按照 本 书 的 步骤 安装 了 MAMP， 就 可 以 直接 使 用 
phpMyAdmin 了 。 

但 是 ， 如 果 设 置 了 MySQL 的 密码 ， 要 想 使 用 phpMyAdmin， 


就 需要 按照 3.5.3 节 介绍 的 内 容 进 行 设置 。 


































































































使 用 phpMyAdmin 


启动 phppMyAdmin 的 方法 有 多 种 ， 其 中 最 简单 的 方法 是 通过 MAMP 的 开始 页 启动 (一 2.2.4 
节 ) 请 在 MAMP 的 页 面 上 点 击 “Open start page”。 








PP 启动 phpMyAdmin 


如 图 Al-1 所 示 ， 在 页 面 项 部 的 菜单 中 选择 “Tools” 一 “phpMyAdmin”。 





Buy MAMP PRO 


phpMyAdmin 选 择 这 个 


SQLite Manager (needs PHP 5.2x or 5.3x 














phpLiteAdmin 


APC (not loaded, 





MySQL 
MySQL can be administer 
To connect to the MySQL 





rffom your own scripts use the following connection parameters: 


图 A1-1 开始 页 
这 样 就 启动 了 phpMyAdmin。 我 们 让 页 面 用 中 文 显示 。 请 在 “语言 -Language” 中 选择 “中 
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文 -Chinese simplified”( 见 图 A1-2 )。 


3 [| [ei 3306 


司 数据 库 膨 SQL 局 状态 可 账户 图 导出 图 导入 Fi Y 更 多 


数据 库 服务 器 
慎 服务 器 连接 排序 规则 加- 服务 器 : localhost via TCPNP | 
utfamb4_unicode_ci 园 | 服务 器 类 型 : MySQL 
| - 服务 器 版 本 : 5.6.34 
MySQL Community Server 
(GPU 
* 协议 版 本 : 10 
外 观 设置 + 用 户 : root@localhost 
惫 语言 - Language 各 2 人 UTF-8 Unicode 


中 文 - Chinese simplified 


恒 二 :|pmanomme 加 | 网 站 服务 叶 




















* 字号: | 82% 区 |]| * Apache/2.2.31 (Win32) DAV/2 

















mod_perV2.0.8 Periv5.16.3 
- 数据 库 客 户 端 版 本 : libmysql - 
5.6.34 
PHP 扩展 : mysqi 局 cun 加 
一 sting 到 




















在 "语言 -Language” 中 选择 "中文 -Chinese simplified” 











A1-2 ” phpMyAdmin 的 初始 页 面 

或 者 直接 在 浏览 器 的 地 址 栏 中 输入 http://localhost/phpMyAdmin 来 启动 phpMyAdmin。 

在 使 用 phpMyAdmin 的 情况 下 ， 即 使 没有 输入 SQL 语句 的 完整 内 容 ， 也 可 以 通过 鼠标 选择 和 
输入 最 低 限 度 需 要 的 内 容 来 创建 查询 。 当 然 ， 我 们 也 可 以 对 执行 的 查询 进行 确认 。 即 使 在 租赁 服务 
器 上 ， 也 有 很 多 无 法 直接 执行 SQL 语句 但 可 以 使 用 phpMyAdmin 的 地 方 。 

如 果 有 可 以 使 用 phpMyAdmin 的 环境 ， 请 一 定 要 尝试 一 下 。 














> 选择 数据 库 和 新 建 数据 库 


选择 数据 库 (use 数据 库 名 ) 和 新 建 数据 库 (CREATE DATABASE 数据 库 名 ) 的 操作 方法 如 
图 A1-3 所 示 。 







fe 06 


日 数据库 | 加 SQL 状态 


















































数据 库 i 
当 创 建新 的 数据 库 
本 performance_schema Ee ~ |] ET 司 Ca 时 ， 请 在 这 里 输入 
= | | 数据 库 名 ， 然 后 点 
数据 库 十 “|z 提 "地 
口 db utf8_general_ci 太 已 复制 喇 检 查 权限 击 “创建 " 按钮 
口 information_schema utf8_general_ci 尽 已 复制 a 检查 权限 
国 mysql utf8_general_ci 业 已 复制 羡 检查 权限 
站 performance_schema utf8_general_ci 坟 已 复制 “检查 权限 点 于 这 里 选 
局 de 择 数 据 库 























MySQL 






































A1-3 ”操作 数据 库 
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区 显示 表 的 记录 、 创 建 表 和 删除 表 


显示 表 的 记录 (SELECT FROM 表 名 )、 创 建 表 ( CREATE 
(DROP TABLE 表 名 ) 的 操作 方法 如 图 Al-4、 图 A1-5 所 示 。 


* 





TABLE 表 名 (...) ) 和 删除 表 


加 
人 大 


SEE 





行 
数 ” 类 型 排序 规则 
4 

口 tblk ”人 帘 园 训 览 月 结 鬼 局 搜索 苇 插 入 赃 清 空 目 型 除 5 InnoDB utr8_general_ci 


21 张 表 总 计 96 InnoDB uti8_general_ci 


+ Dew [Bm 全 


336 
KB 





蚁 打印 骆 Data dictionary 


| 局 六 dB 表 | 




































































点 击 这 里 来 显 当 创 建 表 时 ， 在 这 里 输入 “新 表 名 ” 
示 表 的 记录 和 “字段 数 "， 然 后 点 击 “执行” 按钮 
( 这 里 创建 了 包含 两 个 列 的 表 test ) 




















当 删 除 表 时 ， 选 中 要 
删除 的 表 ， 然 后 点 了 
删除 " 按钮 























A1-4 ”操作 表 












| @ hapynocalhosyphpwyadmivsqlphp?server=ladb=dblatable=tblapos=0 
[locahon/ ocaloont 1 oo x Bl 
phpMyAdmin 
全 国外 者 @ 
近 央 访问 表 收 本 到 









国 浏览 纺 结 鬼 司 SQL 针 搜 索 腾 插 和 局 叶 H 国 导入 


呵 权限 v 更 多 









二 Cument selection does not contain a unique column. Grid edit checkbox, Edit, Copy 
notavailable.、 大 





局 新 建 


and Delete features are 















































由 db1 
| 在 地 第 0-4 行 ( 具 5 行 要 62 缆 00000 秒 . ) 
tb1 SELECT * FROM “tbl' 
3 tb1a 和 
和 tb1b 口 性 裔 分 析 [篇 枉 内 谋 ] [编辑 ] [解析 SQL ] [ 创建 PHP 代码 ][ 刷新 ] 
9 tbic 
| 5: |25 回 过 8 | 在 中 过 | 
name age 
40 
28 
20 
23 
35 
口 显示 全 部 | 行 数 : |25 加 | 过 六 行 : | 在 表 中 搜索 
[ER 人 上 一 























A1-5 显示 了 表 的 记录 





图 A1-6 是 设置 表 text 的 示例 。 其 中 ， 列 empid 具有 自动 连续 编 














号 功能 且 数 据 类 型 为 TINT， 列 
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name 的 数据 类 型 为 VARCHAR (10)。 









€ EE ELT 
phpMyAdmin 
会 避 面目 准 驴 
折 期 访问 表 收 若 夫 
B=- 














Hkh [1 





cournls) 全 







已 新 于 2 
9 名 宁 类 长 性 各 mi 天 序 现 烛 后 空 可 引 
Mb PRIMARY 加 | 
申 W mr 
申 # tbta 
明和 hip 
Bi tbtc 
Bi tpld 
EWtbie 
上 Mbir 
中 tb1g 
唱 贡 um 


name VsckR 加 | [1 无 加 加 回 器 [全 加 口 
表 注 笃 Collation: 存 外 3 学 : 全 


PARTTION demmmon: 可 
Patton by 辐 ( [Bresson orcoumnis] ) 


曲 34 
昌 允 by 
申 放 ok 

i tb2 
朋 M mm 

Lh too 
由 下 
晶 34 mk 





Parttions: 


由 基 ELname 
申 素 Unow 
转注 Laserial 
生 Information_schema 





让 mysql 
覃 9 perfommance_schema 


3 tcst 





洱 
[sy 
渤 
由 
Tu 




















在 这 里 设置 主键 














A1-6 ” 表 的 创建 页 面 


P 插入 记录 、 执 行 任 意 的 SQL 语句 
插入 记录 ( INSERT INTO 表 名 VALUES(...) ) 和 执行 任意 SQL 语句 的 操作 方法 如 图 A1-7 和 
图 A1-8 所 示 。 


cml 
LE) hapynocalnosyphpwyadmivdb structurephp?server=ladb=dblanable=test -dx np- 
父 localhost / localhos/db-x 国 浊 


目 广 结 购 局 SQL 对 搜索 腾 插 入 辐 中 改 导入 问 权限 





虽 关联 视图 











点 击 这 里 











属 = 默 注 
考 ， 名 宝 类 型 排序 规则 性 空 认 释 辣 外 操作 
口 1 empid f int(11) 否 无 。 AUTO_INCREMENT 叶 修 改 @ 删除 v 更 多 
口 2 name ”varchar(10) utfa_ general ci 理 无 她 修改 @ 删除 了 更 多 


人 口 人 2 和 。 施 p 砚 加 庆 览 ”o 修 改 @ 天 除 轧 主妇 国 唯 - 固守 引 





加 打印 ”局 规 划 志 车 构 局 ”到 移动 字段 ”所 Improve table structure 


Es | 1 和 | 于 nanez 加 @) 


键 名 类 型 唯一 紧凑 字段 “基数 排序 规则 空 注释 
咏 编 辑 人 @ 删除 PRIMARY BTREE 是 否 empld 0 A 理 





在 第 |1 。 | 个 守 息 88 家 引 | 【执行 ) 























点 击 这 里 也 可 以 直接 执行 SQL 语 句 











A1-7 通过 上 述 步骤 创建 的 表 test 
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06 > 四 db1 > 融雪 test 
习 浏览 族 结构 局 SQL 有 接 索 有 插入 昌 号 出 忆 导入 


phpMyAdmin 
爹 国 页 门 窜 








字段 类 时 函数 空 值 


empid int(11) E | | | 





























name varchar(10) | 








输入 数据 后 


点 击 这 号 




















回 忽略 


字段 ， 类 型 ”函数 
empid inttD) | 


name varchar(10) | 








































































A1-8 输入 数据 


附录 2 ”常见 问题 的 检查 靖 








如 果 在 运行 MySQL 和 PHP 的 过 程 中 出 现 癌 
个 检查 清单 确认 各 项 目的 状态 。 如 果 有 不 符合 项 


请 试 着 解决 相应 的 问题 ， 问 题解 决 

















题 ， 请 使 用 这 
























































要 求 的 地 方 ， 
吾 再 次 运行 MySQL 和 PHP。 



































安装 过 程 中 出 现 的 问题 


人 一 一 一 一 一 一 一 


”MAMP 安装 失败 


口 使 用 的 是 Windows 7 或 更 高 版 本 的 操作 系统 

口 80 号 端口 和 3306 号 端口 没有 被 其 他 软件 使 用 

口 暂时 禁用 安全 软件 后 尝试 进行 了 安装 

口 没有 在 已 经 安装 了 其 他 版 本 的 MySQL 和 Apache 的 状态 下 安装 MAMP。 安 装 操作 是 在 删除 
所 有 旧 软 件 之 后 进行 的 





S Apache 无 法 运行 
口 确 认 了 在 安装 MAMP 之 前 没有 安装 Apache。 另 外 ， 如 果 已 安装 了 Apache， 保 证 它 已 经 被 和 卸载 
口 在 MAMP 的 启动 画面 中 确认 了 Apache 已 经 运行 (一 2.2.3 节 ) 
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全 在 使 用 MySQL 的 过 程 中 出 现 的 问题 
PP MySQL 无 法 运行 


口 MySQL 不 是 单独 安装 的 ， 而 是 通过 MAMP 统一 安装 的 
口 在 MAMP 的 启动 画面 中 确认 了 MySQL 已 经 运行 
口 正确 设置 了 my.ini ( 一 2.4.2 节 ) ( 恢复 原 设 定 后 可 能 会 正常 运行 ) 





P MySQL 监视 器 无 法 运行 四 


口 正确 安装 了 MAMP 

口 在 MAMP 的 启动 画面 中 确认 了 MySQL 已 经 运行 
口 设 置 了 正确 的 路 径 ， 如 C:\MAMP\binimysql\bin 
口 命令 中 没有 使 用 全 角 字 符 

口 “mysql” 和 “-u 用户 名 ”之 间 输 入 了 半角 空格 
口 “-u” 是 小 写字 母 





> MySQL 监视 器 无 法 运行 @ ( 没有 设置 密码 的 情况 ) 


口 正确 地 输入 了 “mysql -u 用 户 名 ”( mysql -u root 等 ) 
口 没 有 添加 “-p” 选 项 





”MySQL 监视 器 无 法 运行 @) ( 设置 了 密码 的 情况 ) 
口 正 确 地 输入 了 “mysql -u root -p 密码 ” 

口 “-p” 和 “密码 ”之 间 没 有 空格 

口 输入 了 正确 的 密码 

口 在 “-u 用 户 名 ”和 “-p” 之 间 输 入 了 半角 空格 

口 “-u” 和 “-P” 是 小 写字 母 





> MySQL 监视 器 的 显示 中 出 现 乱 码 ， 无 法 输入 中 文 

口 在 my.ini 中 正确 设置 了 字符 编码 ( 一 2.4.2 节 ) 

口 通过 status 显示 的 4 个 字符 编码 是 正确 的 ( 一 3.3.4 节 ) 

口 数据 库 名 、 表 名 和 列 名 中 没有 中 文 ( 如 果 有 ， 试 着 将 其 改 为 半角 字母 ) 
口 当 读 取 文 本 文件 时 ， 该 文件 的 字符 编码 与 正在 使 用 的 字符 编码 相 匹 配 
口 命 令 提示 符 的 字符 编码 是 默认 的 GBK 

口 创建 表 后 ， 没 有 更 改 字 符 编码 
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口 使 用 SoURcE 命令 ( 一 14.2.1 节 ) 执行 的 文本 文件 的 字符 编码 是 GB 2312 
口 使 用 LOAD DATA INFILE 输入 的 文本 文件 的 字符 编码 是 UTF-8 (一 14.1.3 节 ) 
口 当 还 原 转 储 文件 时 ， 还 原 的 文本 文件 的 字符 代码 为 UTF-8 (一 14.4.4 节 ) 





> 准备 创建 存储 函数 时 发 生 错误 
口 在 MySQL 监视 器 中 执行 了 “SET GLOBAL 1og bin trust function creators = 
1;”( 一 12.5.1 节 ) 
P 事务 无 法 正常 运行 
口 开 始 时 执行 了 “START TRANSACTION;” 并 显示 了 Query OK 


口 使 用 SHOW CREATE TABLE 确认 了 表 的 存储 引擎 是 InnoDB 
口 没 有 使 用 会 自动 执行 提交 的 DROP 命令 





> PHP 脚本 无 法 运行 


口 脚本 以 “<?php” 开 头 ， 以 “?>” 结 尾 ， 而 非 “<php?” 

口 没有 拼写 错误 ( prnt 等 ) 

口 没 有 输入 未 用 ' ' 或 " " 括 起 来 的 全 角 字 符 

口 正确 安装 了 MAMP 

口 Apache 处 于 正在 运行 的 状态 。 在 地 址 栏 中 输入 http:WlocalhosVMAMP ， 会 显示 MAMP 的 开 
始 页 面 

口 phpinfo() 能 够 正常 运行 (一 15.8.4 节 ) 

口 PHP 脚本 保存 在 要 发 布 的 文件 夹 中 ( C:\MAMP\htdoc 等 ) 

口 确 认 了 PHP 脚本 的 扩展 名 为 “.php” 

口 Apache 和 PHP 是 通过 MAMP 一 起 安装 的 ( 在 单独 安装 时 如 果 遇 到 问题 ， 试 着 通过 MAMP 
统一 进行 安装 ) 

口 PHP 文件 无 法 通过 双击 直接 打开 





P 无 法 使 用 PHP 脚本 操作 MySQL 

口 MySQL 正在 运行 

口 Apache 正在 运行 

口 phpinfo() 能 够 正常 运行 (一 15.8.4 节 ) 
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口 PHP 脚本 保存 在 要 发 布 的 文件 夹 中 
口 PHP 脚本 的 关键 字 没 有 拼写 错误 

口 脚本 中 不 存在 输入 了 全 角 字 符 等 问题 
口 正确 输入 了 数据 库 名 、 用 户 名 和 密码 





日 期 和 时 间 显 示 不 正常 


OT 
口 在 php.ini 中 设置 了 时 区 (一 15.6.3 节 ) 

P phpMyAdmin 无 法 启动 
口 在 修改 了 mysql 密码 的 情况 下 ，config.inc.php 内 的 密码 也 改 成 了 修改 后 的 字符 ( 一 3.5.3 节 ) 


P phpMyAdmin 上 发 生 错误 
口 正确 完成 了 MySQL 的 my.ini 的 字符 编码 设置 (一 2.4.2 节 ) 


P 字符 乱码 

口 通过 php.ini 正确 设置 了 多 字 节 字符 串 ( 一 15.6.3 节 ) 

口 PHP 脚本 的 源 代码 的 字符 编码 是 UTF-8 

口 在 PHP 脚本 中 插入 了 <meta charset="UTF-8"> (一 17.5 节 ) 


口 在 MySQL 的 表 内 容 出 现 乱 码 的 情况 下 ，MySQL 中 的 字符 编码 原本 是 被 正确 设置 的 
(一 2.4.2 节 ) 





C 在 单独 安装 Apache、PHP、MySQL 的 情况 下 出 现 的 问题 


BP 无 法 单独 安装 Apache、MySQL 和 PHP 

口 安装 了 与 使 用 的 操作 系统 相对 应 的 软件 

口 没 有 在 安装 了 其 他 版 本 的 状态 下 再 进行 安装 ( 全 新 安装 ) 
口试 着 安装 了 其 他 版 本 ( 有 些 版 本 的 安装 程序 存在 缺陷 ) 

口 正确 设置 了 my.ini、php.ini 等 ( 恢复 原 设 定 后 可 能 会 正常 运行 ) 
口 暂 时 禁用 安全 软件 后 尝试 进行 了 安装 





在 本 书 中 ，MySQL 、PHP 和 Apache 相关 内 容 的 介绍 是 以 安装 了 MAMP 为 前 提 的 。 如 果 单 独 
安装 不 顺利 ， 请 使 用 MAMP 对 MySQL 等 统一 进行 安装 。 
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掌握 MySQL 的 最 佳 方法 是 反复 练习 基本 的 SOL 语句 。 如 
果 因为 隔 了 一 段 时 间 没 有 使 用 MySQL 而 忘记 了 基本 语法 ， 可 以 
通过 复习 以 下 操作 流程 来 重 拾 忘 记 的 内 容 。 










































































这 里 ， 我 们 来 复习 一 下 从 创建 数据 库 到 确认 表 这 一 系列 的 操作 流程 ， 具 体 如 下 。 


(QD 启动 MySQL 监视 器 (一 3.3.2 节 )。 
@) 确认 有 哪些 数据 库存 在 (一 4.2.1 节 )。 
@) 决定 使 用 哪 一 个 数据 库 (一 4.3.1 节 )。 
@ 创建 表 ( 一 4.4.3 节 )。 

@) 确认 列 结构 (一 4.6.1 节 )。 

@ 插入 记录 (一 4.7.1 节 )。 

@) 显示 记录 (一 4.8.1 节 )。 








表 世 是 D 股份 有 限 公司 2018 年 第 2 季度 的 销售 信息 表 。 将 各 个 员工 的 信息 输入 到 员工 号 
(empid )、 销 售 额 (sales )、 销 售 月 份 (month ) 3 个 列 中 。 用 户 名 是 “root”， 密 码 是 “root”， 数 据 
库 名 是 “db1”。 
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> 表 tb > 表 tb 的 列 结构 
empid sales month empid VARCHAR (10) 
A103 101 4 sales INT 
A102 54 5 month INT 
A104 181 4 

A101 184 4 

A103 Mla 5 

A101 300 5 

A102 205 6 

A104 93 5 

A103 12 6 

A107 87 6 

















区 4) 启动 MySQL 监视 器 。 
在 命令 提示 符 或 终端 中 输入 以 下 内 容 ， 人 然后 按 Enter 键 。 


> (2 确认 有 哪些 数据 库存 在 。 


P (3) 决定 使 用 哪 一 个 数据 库 。 


区 (4) 创建 表 。 


> (9) 确认 列 结构 。 


附录 3 ”MySQL 基础 练习 | 495 


区 (@ 插入 记录 。 


区 《2Z) 使 用 1 键 显 示 前 一 行 的 内 容 并 进行 修改 ， 然 后 按 Enter 键 ， 循 环 此 操作 。 


> 显示 记录 。 
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点 的 掌握 情况 。 
D) 附录 设 有 “常见 问题 检查 清单 ”， 帮 助 读 者 解决 运行 MySQL 和 PHP 时 经 
常 遇 到 的 问题 。 
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